mirror of https://github.com/renovatebot/renovate
152 lines
3.7 KiB
TypeScript
152 lines
3.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 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 {
|
|
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 isHttpUrl(url: unknown): boolean {
|
|
if (!is.nonEmptyString(url)) {
|
|
return false;
|
|
}
|
|
try {
|
|
const { protocol } = new URL(url);
|
|
return protocol === 'https:' || protocol === 'http:';
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
export function parseUrl(url: URL | string | undefined | null): URL | null {
|
|
if (!url) {
|
|
return null;
|
|
}
|
|
|
|
if (url instanceof URL) {
|
|
return url;
|
|
}
|
|
|
|
try {
|
|
return new URL(url);
|
|
} catch {
|
|
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);
|
|
}
|
|
|
|
/**
|
|
* prefix https:// to hosts with port or path
|
|
*/
|
|
export function massageHostUrl(url: string): string {
|
|
if (!url.includes('://') && url.includes('/')) {
|
|
return 'https://' + url;
|
|
} else if (!url.includes('://') && url.includes(':')) {
|
|
return 'https://' + url;
|
|
} else {
|
|
return url;
|
|
}
|
|
}
|