UNPKG

libmodpm

Version:

Modrinth package manager library

291 lines 19.6 kB
// SPDX-License-Identifier: GPL-3.0-or-later import { HTTPClient } from "../HTTPClient.js"; /** * Represents an error returned by the registry API. * * @final */ class RegistryError extends Error { /** * Error code. */ code; /** * Creates a new registry error. * * @param description Error description. * @param [code] Error code. */ constructor(description, code) { super(description); this.name = new.target.name; this.code = code; } } /** * Provides methods for interacting with the registry via its HTTP API. * * @final */ export class RegistryClient extends HTTPClient { static RegistryError = RegistryError; /** * Base URL for HTTP requests. */ baseUrl; /** * Creates a new registry client. * * @param userAgent User agent string used when making requests to the registry. * @param [token] API authentication token. * @param [baseUrl=new URL("https://api.modrinth.com/v2/")] API authentication token. Requires the following scopes: * - `PROJECT_READ` * - `VERSION_READ` * * Authentication is only needed for accessing private/draft packages and their versions. */ constructor(userAgent, token, baseUrl = new URL("https://api.modrinth.com/v2/")) { super(userAgent, token); this.baseUrl = baseUrl; } /** * Catches 404 errors and returns null. * * @param error Error to try to catch. * @returns `null` if the error is a 404 error. * @throws {@link RegistryClient.RegistryError} If the error is not a 404 error. */ static catch404(error) { if (error instanceof RegistryClient.RegistryError && error.message === "404") return null; throw error; } /** * Retrieves the package associated with the specified ID. * * @param id Package ID. * @returns `null` if the package is not found. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getPackage(id) { return this.fetch(["project", id]) .then(res => res.json()) .catch(RegistryClient.catch404); } /** * Retrieves the ID of the package associated with the specified slug. * * This method returns the ID even for private/draft packages, without requiring authentication. * * @param slug Package slug. * @returns `null` if the package is not found. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getPackageId(slug) { return this.fetch(["project", slug, "check"]) .then(res => res.json()) .then(json => json.id) .catch(RegistryClient.catch404); } /** * Retrieves packages matching the specified search query and facet filters. * * @param [query] Search query. * @param [facets] Facets to filter by. * @param [sort] Sort order. * @param [offset] Offset into the search. * @param [limit] Maximum number of results to return. * * @see https://docs.modrinth.com/api/operations/searchprojects/ Search projects | Modrinth Documentation */ async search(query, facets, sort, offset, limit = 20) { const queryParams = new URLSearchParams(); if (query !== undefined) queryParams.set("query", query); if (facets !== undefined) queryParams.set("facets", JSON.stringify(facets)); if (sort !== undefined) queryParams.set("sort", sort); if (offset !== undefined) queryParams.set("offset", offset.toString()); queryParams.set("limit", limit.toString()); const body = await this.fetch(["search"], {}, queryParams).then(res => res.text()); return JSON.parse(body, (_, value) => { if (typeof value === "string" && /^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}.\d{3}Z$/.test(value)) return new Date(value); return value; }); } /** * Retrieves the version associated with the specified version ID. * * @param id Version ID. * @returns `null` if the version is not found. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getVersion(id) { return this.fetch(["version", id]) .then(res => res.json()) .catch(RegistryClient.catch404); } /** * Retrieves the versions associated with the specified package. * * Filters: * - `loaders` — restricts versions to those compatible with the specified loaders. * - `game_versions` — restricts versions to those compatible with the specified game versions. * - `version_type` — restricts versions to those of the specified type (release channel). * * @param pkg Package ID. * @param [filters] Filters to apply. * @returns List of matching versions, sorted in descending order, with the latest version first. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async listVersions(pkg, filters = {}) { const params = new URLSearchParams(); if (filters.loaders !== undefined) params.append("loaders", JSON.stringify(filters.loaders)); if (filters.game_versions !== undefined) params.append("game_versions", JSON.stringify(filters.game_versions)); if (filters.version_type !== undefined) params.append("version_type", JSON.stringify(filters.version_type)); return this.fetch(["project", pkg, "version"], {}, params) .then(res => res.json()) .catch(RegistryClient.catch404); } /** * Retrieves the packages associated with the specified IDs. * * @param ids Package IDs. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getPackages(ids) { return this.fetch(["projects"], {}, new URLSearchParams({ ids: JSON.stringify(ids), })).then(res => res.json()); } /** * Retrieves the version associated with the specified file hash. * * @param hash SHA-512 hash of the file, encoded as a hex string. * @returns `null` if the version is not found. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getVersionByHash(hash) { return this.fetch(["version_file", hash], {}, new URLSearchParams({ algorithm: "sha512", })) .then(res => res.json()) .catch(RegistryClient.catch404); } /** * Retrieves the versions associated with the specified hashes. * * @param hashes SHA-512 hashes of the files, encoded as a hex strings. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getVersionsByHashes(hashes) { return this.fetch(["version_files"], { method: "POST", headers: { "Content-Type": "application/json", }, body: JSON.stringify({ hashes, algorithm: "sha512", }), }).then(res => res.json()) .then((json) => Object.values(json)); } /** * Retrieves the versions associated with the specified version IDs. * * @param ids Version IDs. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getVersions(ids) { return this.fetch(["versions"], {}, new URLSearchParams({ ids: JSON.stringify(ids), })).then(res => res.json()); } /** * Retrieves the version associated with the specified package and version number or ID. * * @param pkg Package ID. * @param version Version number or ID. * @returns `null` if the package or version is not found. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getVersionByNumber(pkg, version) { return this.fetch(["project", pkg, "version", version]) .then(res => res.json()) .catch(err => { if (err instanceof RegistryClient.RegistryError && err.message === "404") return null; throw err; }); } /** * Retrieves the latest versions associated with the specified hashes. * * Filters: * - `loaders` — restricts versions to those compatible with the specified loaders. * - `game_versions` — restricts versions to those compatible with the specified game versions. * - `version_type` — restricts versions to those of the specified type (release channel). * * @param hashes SHA-512 hashes of the files, encoded as a hex strings. * @param [filters] Filters to apply. * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getLatestVersions(hashes, filters = {}) { return this.fetch(["version_files", "update"], { method: "POST", headers: { "Content-Type": "application/json", }, body: JSON.stringify({ hashes, algorithm: "sha512", loaders: filters.loaders, game_versions: filters.game_versions, version_type: filters.version_type, }), }).then(res => res.json()); } /** * Retrieves all loaders supported by the registry. * * @throws {@link RegistryClient.RegistryError} If the request fails. * @throws {@link !TypeError} If fetching fails. */ async getLoaders() { return this.fetch(["tag", "loader"]) .then(res => res.json()) .then((json) => json.map((l) => l.name)); } async createError(res) { if ((res.headers.get("Content-Type")?.startsWith("application/json") ?? false) && res.body !== null) { const { description, error } = await res.json(); return new RegistryError(description, error); } return new RegistryError(res.status.toString()); } // only adding an overload… sorry @final 😔 async fetch(url, options, query, retries = 3, remainingRetries = retries) { if (Array.isArray(url)) return super.fetch(new URL(url.map(globalThis.encodeURIComponent).join("/"), this.baseUrl), options, query, remainingRetries); return super.fetch(url, options, query, remainingRetries); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RegistryClient.js","sourceRoot":"","sources":["../../src/registry/RegistryClient.ts"],"names":[],"mappings":"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"}