ufo
URL utils for humans.
URL utils for humans.
Install
Install using npm or your favourite package manager:
Install package:
# npm
npm install ufo
# yarn
yarn add ufo
# pnpm
pnpm install ufo
# bun
bun install ufo
Import utils:
// ESM
import { normalizeURL, joinURL } from "ufo";
// CommonJS
const { normalizeURL, joinURL } = require("ufo");
// Deno
import { parseURL } from "https://unpkg.com/ufo/dist/index.mjs";
Encoding Utils
decode(text)
Decodes text using decodeURIComponent. Returns the original text if it fails.
decodePath(text)
Decodes path section of URL (consistent with encodePath for slash encoding).
decodeQueryKey(text)
Decodes query key (consistent with encodeQueryKey for plus encoding).
decodeQueryValue(text)
Decodes query value (consistent with encodeQueryValue for plus encoding).
encode(text)
Encodes characters that need to be encoded in the path, search and hash sections of the URL.
encodeHash(text)
Encodes characters that need to be encoded in the hash section of the URL.
encodeHost(name)
Encodes hostname with punycode encoding.
encodeParam(text)
Encodes characters that need to be encoded in the path section of the URL as a param. This function encodes everything encodePath does plus the slash (/) character.
encodePath(text)
Encodes characters that need to be encoded in the path section of the URL.
encodeQueryKey(text)
Encodes characters that need to be encoded for query values in the query section of the URL and also encodes the = character.
encodeQueryValue(input)
Encodes characters that need to be encoded for query values in the query section of the URL.
Parsing Utils
parseAuth(input)
Takes a string of the form username:password and returns an object with the username and password decoded.
parseFilename(input, opts?: { strict? })
Parses a URL and returns last segment in path as filename.
If { strict: true } is passed as the second argument, it will only return the last segment only if ending with an extension.
Example:
// Result: filename.ext
parseFilename("http://example.com/path/to/filename.ext");
// Result: undefined
parseFilename("/path/to/.hidden-file", { strict: true });
parseHost(input)
Takes a string, and returns an object with two properties: hostname and port.
Example:
parseHost("foo.com:8080");
// { hostname: 'foo.com', port: '8080' }
parsePath(input)
Splits the input string into three parts, and returns an object with those three parts.
Example:
parsePath("http://foo.com/foo?test=123#token");
// { pathname: 'http://foo.com/foo', search: '?test=123', hash: '#token' }
parseURL(input, defaultProto?)
Takes a URL string and returns an object with the URL's protocol, auth, host, pathname, search, and hash.
Example:
parseURL("http://foo.com/foo?test=123#token");
// { protocol: 'http:', auth: '', host: 'foo.com', pathname: '/foo', search: '?test=123', hash: '#token' }
parseURL("foo.com/foo?test=123#token");
// { pathname: 'foo.com/foo', search: '?test=123', hash: '#token' }
parseURL("foo.com/foo?test=123#token", "https://");
// { protocol: 'https:', auth: '', host: 'foo.com', pathname: '/foo', search: '?test=123', hash: '#token' }
stringifyParsedURL(parsed)
Takes a ParsedURL object and returns the stringified URL.
Example:
const obj = parseURL("http://foo.com/foo?test=123#token");
obj.host = "bar.com";
stringifyParsedURL(obj); // "http://bar.com/foo?test=123#token"
Query Utils
encodeQueryItem(key, value)
Encodes a pair of key and value into a url query string value.
If the value is an array, it will be encoded as multiple key-value pairs with the same key.
Example:
encodeQueryItem('message', 'Hello World')
// 'message=Hello+World'
encodeQueryItem('tags', ['javascript', 'web', 'dev'])
// 'tags=javascript&tags=web&tags=dev'
parseQuery(parametersString)
Parses and decodes a query string into an object.
The input can be a query string with or without the leading ?.
Example:
parseQuery("?foo=bar&baz=qux");
// { foo: "bar", baz: "qux" }
parseQuery("tags=javascript&tags=web&tags=dev");
// { tags: ["javascript", "web", "dev"] }
stringifyQuery(query)
Stringfies and encodes a query object into a query string.
Example:
stringifyQuery({ foo: 'bar', baz: 'qux' })
// 'foo=bar&baz=qux'
stringifyQuery({ foo: 'bar', baz: undefined })
// 'foo=bar'
Utils
$URL()
cleanDoubleSlashes(input)
Removes double slashes from the URL.
Example:
cleanDoubleSlashes("//foo//bar//"); // "/foo/bar/"
cleanDoubleSlashes("http://example.com/analyze//http://localhost:3000//");
// Returns "http://example.com/analyze/http://localhost:3000/"
filterQuery(input, predicate)
Removes the query section of the URL.
Example:
filterQuery("/foo?bar=1&baz=2", (key) => key !== "bar"); // "/foo?baz=2"
getQuery(input)
Parses and decodes the query object of an input URL into an object.
Example:
getQuery("http://foo.com/foo?test=123&unicode=%E5%A5%BD");
// { test: "123", unicode: "好" }
hasLeadingSlash(input)
Checks if the input has a leading slash (e.g. /foo).
hasProtocol(inputString, opts)
Checks if the input has a protocol.
You can use { acceptRelative: true } to accept relative URLs as valid protocol.
Example:
hasProtocol('https://example.com'); // true
hasProtocol("//example.com"); // false
hasProtocol('//example.com', { acceptRelative: true });  // true
hasProtocol("ftp://example.com"); // true
hasProtocol('data:text/plain'); // true
hasProtocol('data:text/plain', { strict: true }); // false
hasTrailingSlash(input, respectQueryAndFragment?)
Checks if the input has a trailing slash.
Example:
hasTrailingSlash("/foo/"); // true
hasTrailingSlash("/foo"); // false
hasTrailingSlash("/foo?query=true", true); // false
hasTrailingSlash("/foo/?query=true", true); // true
isEmptyURL(url)
Checks if the input URL is empty or /.
isEqual(a, b, options)
Checks if two paths are equal regardless of encoding, trailing slash, and leading slash differences.
You can make slash check strict by setting { trailingSlash: true, leadingSlash: true } as options.
You can make encoding check strict by setting { encoding: true } as options.
Example:
isEqual("/foo", "foo"); // true
isEqual("foo/", "foo"); // true
isEqual("/foo bar", "/foo%20bar"); // true
// Strict compare
isEqual("/foo", "foo", { leadingSlash: true }); // false
isEqual("foo/", "foo", { trailingSlash: true }); // false
isEqual("/foo bar", "/foo%20bar", { encoding: true }); // false
isNonEmptyURL(url)
Checks if the input URL is neither empty nor /.
isRelative(inputString)
Check if a path starts with ./ or ../.
Example:
isRelative("./foo"); // true
isSamePath(p1, p2)
Check if two paths are equal or not. Trailing slash and encoding are normalized before comparison.
Example:
isSamePath("/foo", "/foo/"); // true
isScriptProtocol(protocol?)
Checks if the input protocol is any of the dangerous blob:, data:, javascript: or vbscript: protocols.
Example:
isScriptProtocol("javascript:alert(1)"); // true
isScriptProtocol("data:text/html,hello"); // true
isScriptProtocol("blob:hello"); // true
isScriptProtocol("vbscript:alert(1)"); // true
isScriptProtocol("https://example.com"); // false
joinRelativeURL()
Joins multiple URL segments into a single URL and also handles relative paths with ./ and ../.
Example:
joinRelativeURL("/a", "../b", "./c"); // "/b/c"
joinURL(base)
Joins multiple URL segments into a single URL.
Example:
joinURL("a", "/b", "/c"); // "a/b/c"
normalizeURL(input)
Normalizes the input URL:
- Ensures the URL is properly encoded - Ensures pathname starts with a slash - Preserves protocol/host if provided
Example:
normalizeURL("test?query=123 123#hash, test");
// Returns "test?query=123%20123#hash,%20test"
normalizeURL("http://localhost:3000");
// Returns "http://localhost:3000"
resolveURL(base)
Resolves multiple URL segments into a single URL.
Example:
resolveURL("http://foo.com/foo?test=123#token", "bar", "baz");
// Returns "http://foo.com/foo/bar/baz?test=123#token"
withBase(input, base)
Ensures the URL or pathname starts with base.
If input already starts with base, it will not be added again.
Example:
withBase("/foo/bar", "/foo"); // "/foo/bar"
withBase("/foo/bar", "/baz"); // "/baz/foo/bar"
withFragment(input, hash)
Adds or replaces the fragment section of the URL.
Example:
withFragment("/foo", "bar"); // "/foo#bar"
withFragment("/foo#bar", "baz"); // "/foo#baz"
withFragment("/foo#bar", ""); // "/foo"
withHttp(input)
Adds or replaces the URL protocol to http://.
Example:
withHttp("https://example.com"); // http://example.com
withHttps(input)
Adds or replaces the URL protocol to https://.
Example:
withHttps("http://example.com"); // https://example.com
withLeadingSlash(input)
Ensures the URL or pathname has a leading slash.
Example:
withLeadingSlash("foo"); // "/foo"
withoutBase(input, base)
Removes the base from the URL or pathname.
If input does not start with base, it will not be removed.
Example:
withoutBase("/foo/bar", "/foo"); // "/bar"
withoutFragment(input)
Removes the fragment section from the URL.
Example:
withoutFragment("http://example.com/foo?q=123#bar")
// Returns "http://example.com/foo?q=123"
withoutHost(input)
Removes the host from the URL while preserving everything else.
Example:
withoutHost("http://example.com/foo?q=123#bar")
// Returns "/foo?q=123#bar"
withoutLeadingSlash(input)
Removes leading slash from the URL or pathname.
Example:
withoutLeadingSlash("/foo"); // "foo"
withoutProtocol(input)
Removes the protocol from the input.
Example:
withoutProtocol("http://example.com"); // "example.com"
withoutTrailingSlash(input, respectQueryAndFragment?)
Removes the trailing slash from the URL or pathname.
If second argument is true, it will only remove the trailing slash if it's not part of the query or fragment with cost of more expensive operations.
Example:
withoutTrailingSlash("/foo/"); // "/foo"
withoutTrailingSlash("/path/?query=true", true); // "/path?query=true"
withProtocol(input, protocol)
Adds or replaces protocol of the input URL.
Example:
withProtocol("http://example.com", "ftp://"); // "ftp://example.com"
withQuery(input, query)
Add/Replace the query section of the URL.
Example:
withQuery("/foo?page=a", { token: "secret" }); // "/foo?page=a&token=secret"
withTrailingSlash(input, respectQueryAndFragment?)
Ensures the URL ends with a trailing slash.
If second argument is true, it will only add the trailing slash if it's not part of the query or fragment with cost of more expensive operation.
Example:
withTrailingSlash("/foo"); // "/foo/"
withTrailingSlash("/path?query=true", true); // "/path/?query=true"
License
Special thanks to Eduardo San Martin Morote (posva) for encoding utilities