import is from '@sindresorhus/is'; // eslint-disable-next-line no-restricted-imports import _parseLinkHeader from 'parse-link-header'; import urlJoin from 'url-join'; import { logger } from '../logger'; import { regEx } from './regex'; export function joinUrlParts(...parts: string[]): string { return urlJoin(...parts); } export function ensurePathPrefix(url: string, prefix: string): string { const parsed = new URL(url); const fullPath = parsed.pathname + parsed.search; if (fullPath.startsWith(prefix)) { return url; } return parsed.origin + prefix + fullPath; } export function ensureTrailingSlash(url: string): string { return url.replace(/\/?$/, '/'); // TODO #12875 adds slash at the front when re2 is used } export function trimTrailingSlash(url: string): string { return url.replace(regEx(/\/+$/), ''); } export function trimLeadingSlash(path: string): string { return path.replace(/^\/+/, ''); } export function trimSlashes(path: string): string { return trimLeadingSlash(trimTrailingSlash(path)); } /** * Resolves an input path against a base URL * * @param baseUrl - base URL to resolve against * @param input - input path (if this is a full URL, it will be returned) */ export function resolveBaseUrl(baseUrl: string, input: string | URL): string { const inputString = input.toString(); let host; let pathname; try { ({ host, pathname } = new URL(inputString)); } catch (e) { pathname = inputString; } return host ? inputString : urlJoin(baseUrl, pathname || ''); } /** * Replaces the path of a URL with a new path * * @param baseUrl - source URL * @param path - replacement path (if this is a full URL, it will be returned) */ export function replaceUrlPath(baseUrl: string | URL, path: string): string { if (parseUrl(path)) { return path; } const { origin } = is.string(baseUrl) ? new URL(baseUrl) : baseUrl; return urlJoin(origin, path); } export function getQueryString(params: Record): string { const usp = new URLSearchParams(); for (const [k, v] of Object.entries(params)) { if (is.array(v)) { for (const item of v) { // TODO: fix me? // eslint-disable-next-line @typescript-eslint/no-base-to-string usp.append(k, item.toString()); } } else { usp.append(k, v.toString()); } } return usp.toString(); } export function validateUrl(url?: string, httpOnly = true): boolean { if (!url) { return false; } try { const { protocol } = new URL(url); return httpOnly ? !!protocol.startsWith('http') : !!protocol; } catch (err) { return false; } } export function parseUrl(url: string | undefined | null): URL | null { if (!url) { return null; } try { return new URL(url); } catch (err) { return null; } } /** * Tries to create an URL object from either a full URL string or a hostname * @param url either the full url or a hostname * @returns an URL object or null */ export function createURLFromHostOrURL(url: string): URL | null { return parseUrl(url) ?? parseUrl(`https://${url}`); } export type LinkHeaderLinks = _parseLinkHeader.Links; export function parseLinkHeader( linkHeader: string | null | undefined ): LinkHeaderLinks | null { if (!is.nonEmptyString(linkHeader)) { return null; } if (linkHeader.length > 2000) { logger.warn({ linkHeader }, 'Link header too long.'); return null; } return _parseLinkHeader(linkHeader); }