libmodpm
Version:
Modrinth package manager library
291 lines • 19.6 kB
JavaScript
// 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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"}