Skip to content

Commit

Permalink
GraphQLError: fix empty locations if error got nodes without locati…
Browse files Browse the repository at this point in the history
…ons (#3332)
  • Loading branch information
IvanGoncharov committed Oct 26, 2021
1 parent bbb1af5 commit 8ba5c56
Show file tree
Hide file tree
Showing 3 changed files with 40 additions and 41 deletions.
68 changes: 28 additions & 40 deletions src/error/GraphQLError.js
Expand Up @@ -87,51 +87,33 @@ export class GraphQLError extends Error {
super(message);

// Compute list of blame nodes.
const _nodes = Array.isArray(nodes)
? nodes.length !== 0
? nodes
: undefined
: nodes
? [nodes]
: undefined;
const _nodes = undefinedIfEmpty(
Array.isArray(nodes) ? nodes : nodes ? [nodes] : undefined,
);

let nodeLocations = [];
for (const { loc } of _nodes ?? []) {
if (loc != null) {
nodeLocations.push(loc);
}
}
nodeLocations = undefinedIfEmpty(nodeLocations);

// Compute locations in the source for the given nodes/positions.
let _source = source;
if (!_source && _nodes) {
_source = _nodes[0].loc?.source;
}
const _source = source ?? nodeLocations?.[0].source;

let _positions = positions;
if (!_positions && _nodes) {
_positions = _nodes.reduce((list, node) => {
if (node.loc) {
list.push(node.loc.start);
}
return list;
}, []);
}
if (_positions && _positions.length === 0) {
_positions = undefined;
}
const _positions = positions ?? nodeLocations?.map((loc) => loc.start);

let _locations;
if (positions && source) {
_locations = positions.map((pos) => getLocation(source, pos));
} else if (_nodes) {
_locations = _nodes.reduce((list, node) => {
if (node.loc) {
list.push(getLocation(node.loc.source, node.loc.start));
}
return list;
}, []);
}
const _locations =
positions && source
? positions.map((pos) => getLocation(source, pos))
: nodeLocations?.map((loc) => getLocation(loc.source, loc.start));

let _extensions = extensions;
if (_extensions == null && originalError != null) {
const originalExtensions = originalError.extensions;
if (isObjectLike(originalExtensions)) {
_extensions = originalExtensions;
}
let _extensions = extensions ?? undefined;

const originalExtensions = originalError?.extensions;
if (isObjectLike(originalExtensions)) {
_extensions = originalExtensions;
}

Object.defineProperties((this: any), {
Expand Down Expand Up @@ -218,6 +200,12 @@ export class GraphQLError extends Error {
}
}

function undefinedIfEmpty<T>(
array: $ReadOnlyArray<T> | void,
): $ReadOnlyArray<T> | void {
return array === undefined || array.length === 0 ? undefined : array;
}

/**
* Prints a GraphQLError to a string, representing useful location information
* about the error's position in the source.
Expand Down
12 changes: 12 additions & 0 deletions src/error/__tests__/GraphQLError-test.js
Expand Up @@ -96,6 +96,18 @@ describe('GraphQLError', () => {
});
});

it('converts node without location to undefined source, positions and locations', () => {
const documentNode = parse('{ foo }', { noLocation: true });

const e = new GraphQLError('msg', documentNode);
expect(e).to.deep.include({
nodes: [documentNode],
source: undefined,
positions: undefined,
locations: undefined,
});
});

it('converts source and positions to locations', () => {
const e = new GraphQLError('msg', null, source, [6]);
expect(e).to.have.property('source', source);
Expand Down
1 change: 0 additions & 1 deletion src/validation/__tests__/validation-test.js
Expand Up @@ -136,7 +136,6 @@ describe('Validate: Limit maximum number of validation errors', () => {
function invalidFieldError(fieldName: string) {
return {
message: `Cannot query field "${fieldName}" on type "QueryRoot".`,
locations: [],
};
}

Expand Down

0 comments on commit 8ba5c56

Please sign in to comment.