mirror of
https://github.com/packwiz/packwiz.git
synced 2025-04-19 21:16:30 +02:00
206 lines
5.5 KiB
Go
206 lines
5.5 KiB
Go
package modrinth
|
|
|
|
import (
|
|
modrinthApi "codeberg.org/jmansfield/go-modrinth/modrinth"
|
|
"errors"
|
|
"github.com/packwiz/packwiz/cmd"
|
|
"github.com/packwiz/packwiz/core"
|
|
"github.com/spf13/cobra"
|
|
"github.com/spf13/viper"
|
|
"github.com/unascribed/FlexVer/go/flexver"
|
|
"golang.org/x/exp/slices"
|
|
"net/http"
|
|
"net/url"
|
|
"regexp"
|
|
)
|
|
|
|
var modrinthCmd = &cobra.Command{
|
|
Use: "modrinth",
|
|
Aliases: []string{"mr"},
|
|
Short: "Manage modrinth-based mods",
|
|
}
|
|
|
|
var mrDefaultClient = modrinthApi.NewClient(&http.Client{})
|
|
|
|
func init() {
|
|
cmd.Add(modrinthCmd)
|
|
core.Updaters["modrinth"] = mrUpdater{}
|
|
|
|
mrDefaultClient.UserAgent = core.UserAgent
|
|
}
|
|
|
|
func getProjectIdsViaSearch(query string, versions []string) ([]*modrinthApi.SearchResult, error) {
|
|
facets := make([]string, 0)
|
|
for _, v := range versions {
|
|
facets = append(facets, "versions:"+v)
|
|
}
|
|
|
|
res, err := mrDefaultClient.Projects.Search(&modrinthApi.SearchOptions{
|
|
Limit: 5,
|
|
Index: "relevance",
|
|
Query: query,
|
|
})
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return res.Hits, nil
|
|
}
|
|
|
|
var urlRegexes = [...]*regexp.Regexp{
|
|
// Slug/version number regex from https://github.com/modrinth/labrinth/blob/1679a3f844497d756d0cf272c5374a5236eabd42/src/util/validate.rs#L8
|
|
regexp.MustCompile("^https?://modrinth\\.com/(?P<projectType>[^/]+)/(?P<slug>[a-zA-Z0-9!@$()`.+,_\"-]{3,64})(?:/version/(?P<version>[a-zA-Z0-9!@$()`.+,_\"-]{1,32}))?"),
|
|
// Version/project IDs are more restrictive: [a-zA-Z0-9]+ (base62)
|
|
regexp.MustCompile("^https?://cdn\\.modrinth\\.com/data/(?P<slug>[a-zA-Z0-9]+)/versions/(?P<versionID>[a-zA-Z0-9]+)/(?P<filename>[^/]+)$"),
|
|
regexp.MustCompile("^(?P<slug>[a-zA-Z0-9!@$()`.+,_\"-]{3,64})$"),
|
|
}
|
|
|
|
const slugRegexIdx = 2
|
|
|
|
var projectTypes = []string{
|
|
"mod", "plugin", "datapack", "shader", "resourcepack", "modpack",
|
|
}
|
|
|
|
func parseSlugOrUrl(input string, slug *string, version *string, versionID *string, filename *string) (parsedSlug bool, err error) {
|
|
for regexIdx, r := range urlRegexes {
|
|
matches := r.FindStringSubmatch(input)
|
|
if matches != nil {
|
|
if i := r.SubexpIndex("projectType"); i >= 0 {
|
|
if !slices.Contains(projectTypes, matches[i]) {
|
|
err = errors.New("unknown project type: " + matches[i])
|
|
return
|
|
}
|
|
}
|
|
if i := r.SubexpIndex("slug"); i >= 0 {
|
|
*slug = matches[i]
|
|
}
|
|
if i := r.SubexpIndex("version"); i >= 0 {
|
|
*version = matches[i]
|
|
}
|
|
if i := r.SubexpIndex("versionID"); i >= 0 {
|
|
*versionID = matches[i]
|
|
}
|
|
if i := r.SubexpIndex("filename"); i >= 0 {
|
|
var parsed string
|
|
parsed, err = url.PathUnescape(matches[i])
|
|
if err != nil {
|
|
return
|
|
}
|
|
*filename = parsed
|
|
}
|
|
parsedSlug = regexIdx == slugRegexIdx
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func getLatestVersion(projectID string, pack core.Pack) (*modrinthApi.Version, error) {
|
|
mcVersion, err := pack.GetMCVersion()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
gameVersions := append([]string{mcVersion}, viper.GetStringSlice("acceptable-game-versions")...)
|
|
|
|
result, err := mrDefaultClient.Versions.ListVersions(projectID, modrinthApi.ListVersionsOptions{
|
|
GameVersions: gameVersions,
|
|
Loaders: pack.GetLoaders(),
|
|
// TODO: change based on project type? or just add iris/optifine/datapack/vanilla/minecraft as default loaders
|
|
// TODO: add "datapack" as a loader *if* a path to store datapacks in is configured?
|
|
})
|
|
|
|
if len(result) == 0 {
|
|
return nil, errors.New("no valid versions found")
|
|
}
|
|
|
|
latestValidVersion := result[0]
|
|
for _, v := range result[1:] {
|
|
// Use FlexVer to compare versions
|
|
compare := flexver.Compare(*v.VersionNumber, *latestValidVersion.VersionNumber)
|
|
|
|
if compare == 0 {
|
|
// Prefer Quilt over Fabric (Modrinth backend handles filtering)
|
|
if slices.Contains(v.Loaders, "quilt") && !slices.Contains(latestValidVersion.Loaders, "quilt") {
|
|
latestValidVersion = v
|
|
continue
|
|
}
|
|
|
|
// FlexVer comparison is equal, compare date instead
|
|
// TODO: flag to force comparing by date?
|
|
if v.DatePublished.After(*latestValidVersion.DatePublished) {
|
|
latestValidVersion = v
|
|
}
|
|
} else if compare > 0 {
|
|
latestValidVersion = v
|
|
}
|
|
}
|
|
|
|
return latestValidVersion, nil
|
|
}
|
|
|
|
func getSide(mod *modrinthApi.Project) string {
|
|
server := shouldDownloadOnSide(*mod.ServerSide)
|
|
client := shouldDownloadOnSide(*mod.ClientSide)
|
|
|
|
if server && client {
|
|
return core.UniversalSide
|
|
} else if server {
|
|
return core.ServerSide
|
|
} else if client {
|
|
return core.ClientSide
|
|
} else {
|
|
return ""
|
|
}
|
|
}
|
|
|
|
func shouldDownloadOnSide(side string) bool {
|
|
return side == "required" || side == "optional"
|
|
}
|
|
|
|
func getBestHash(v *modrinthApi.File) (string, string) {
|
|
// Try preferred hashes first; SHA1 is first as it is required for Modrinth pack exporting
|
|
val, exists := v.Hashes["sha1"]
|
|
if exists {
|
|
return "sha1", val
|
|
}
|
|
val, exists = v.Hashes["sha512"]
|
|
if exists {
|
|
return "sha512", val
|
|
}
|
|
val, exists = v.Hashes["sha256"]
|
|
if exists {
|
|
return "sha256", val
|
|
}
|
|
val, exists = v.Hashes["murmur2"] // (not defined in Modrinth pack spec, use with caution)
|
|
if exists {
|
|
return "murmur2", val
|
|
}
|
|
|
|
//none of the preferred hashes are present, just get the first one
|
|
for key, val := range v.Hashes {
|
|
return key, val
|
|
}
|
|
|
|
//No hashes were present
|
|
return "", ""
|
|
}
|
|
|
|
func getInstalledProjectIDs(index *core.Index) []string {
|
|
var installedProjects []string
|
|
for _, modPath := range index.GetAllMods() {
|
|
mod, err := core.LoadMod(modPath)
|
|
if err == nil {
|
|
data, ok := mod.GetParsedUpdateData("modrinth")
|
|
if ok {
|
|
updateData, ok := data.(mrUpdateData)
|
|
if ok {
|
|
if len(updateData.ProjectID) > 0 {
|
|
installedProjects = append(installedProjects, updateData.ProjectID)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return installedProjects
|
|
}
|