@graphql-tools/utils
Version:
Common package containing utils and types for GraphQL tools
135 lines (134 loc) • 3.91 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.asArray = void 0;
exports.isUrl = isUrl;
exports.isDocumentString = isDocumentString;
exports.isValidPath = isValidPath;
exports.compareStrings = compareStrings;
exports.nodeToString = nodeToString;
exports.compareNodes = compareNodes;
exports.isSome = isSome;
exports.assertSome = assertSome;
const graphql_1 = require("graphql");
const URL_REGEXP = /^(https?|wss?|file):\/\//;
/**
* Checks if the given string is a valid URL.
*
* @param str - The string to validate as a URL
* @returns A boolean indicating whether the string is a valid URL
*
* @remarks
* This function first attempts to use the `URL.canParse` method if available.
* If not, it falls back to creating a new `URL` object to validate the string.
*/
function isUrl(str) {
if (typeof str !== 'string') {
return false;
}
if (!URL_REGEXP.test(str)) {
return false;
}
if (URL.canParse) {
return URL.canParse(str);
}
try {
const url = new URL(str);
return !!url;
}
catch (e) {
return false;
}
}
const asArray = (fns) => (Array.isArray(fns) ? fns : fns ? [fns] : []);
exports.asArray = asArray;
const invalidDocRegex = /\.[a-z0-9]+$/i;
/**
* Determines if a given input is a valid GraphQL document string.
*
* @param str - The input to validate as a GraphQL document
* @returns A boolean indicating whether the input is a valid GraphQL document string
*
* @remarks
* This function performs several validation checks:
* - Ensures the input is a string
* - Filters out strings with invalid document extensions
* - Excludes URLs
* - Attempts to parse the string as a GraphQL document
*
* @throws {Error} If the document fails to parse and is empty except GraphQL comments
*/
function isDocumentString(str) {
if (typeof str !== 'string') {
return false;
}
// XXX: is-valid-path or is-glob treat SDL as a valid path
// (`scalar Date` for example)
// this why checking the extension is fast enough
// and prevent from parsing the string in order to find out
// if the string is a SDL
if (invalidDocRegex.test(str) || isUrl(str)) {
return false;
}
try {
(0, graphql_1.parse)(str);
return true;
}
catch (e) {
if (!e.message.includes('EOF') &&
str.replace(/(\#[^*]*)/g, '').trim() !== '' &&
str.includes(' ')) {
throw new Error(`Failed to parse the GraphQL document. ${e.message}\n${str}`);
}
}
return false;
}
const invalidPathRegex = /[‘“!%^<>`\n]/;
/**
* Checkes whether the `str` contains any path illegal characters.
*
* A string may sometimes look like a path but is not (like an SDL of a simple
* GraphQL schema). To make sure we don't yield false-positives in such cases,
* we disallow new lines in paths (even though most Unix systems support new
* lines in file names).
*/
function isValidPath(str) {
return typeof str === 'string' && !invalidPathRegex.test(str);
}
function compareStrings(a, b) {
if (String(a) < String(b)) {
return -1;
}
if (String(a) > String(b)) {
return 1;
}
return 0;
}
function nodeToString(a) {
let name;
if ('alias' in a) {
name = a.alias?.value;
}
if (name == null && 'name' in a) {
name = a.name?.value;
}
if (name == null) {
name = a.kind;
}
return name;
}
function compareNodes(a, b, customFn) {
const aStr = nodeToString(a);
const bStr = nodeToString(b);
if (typeof customFn === 'function') {
return customFn(aStr, bStr);
}
return compareStrings(aStr, bStr);
}
function isSome(input) {
return input != null;
}
function assertSome(input, message = 'Value should be something') {
if (input == null) {
throw new Error(message);
}
}
;