renovate/lib/util/url.ts
renovate[bot] fe739befbf
chore(deps): update linters to v6 (major) (#23325)
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
Co-authored-by: Michael Kriese <michael.kriese@visualon.de>
2023-08-02 15:07:49 +00:00

134 lines
3.4 KiB
TypeScript

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, any>): string {
const usp = new URLSearchParams();
for (const [k, v] of Object.entries(params)) {
if (is.array<object>(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);
}