renovate/lib/util/url.ts

108 lines
2.7 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 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 || '');
}
export function getQueryString(params: Record<string, any>): string {
const usp = new URLSearchParams();
for (const [k, v] of Object.entries(params)) {
if (Array.isArray(v)) {
for (const item of v) {
usp.append(k, item.toString());
}
} else {
usp.append(k, v.toString());
}
}
const res = usp.toString();
return res;
}
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);
}