UNPKG

graphql

Version:

A Query Language and Runtime which can target any service.

254 lines (231 loc) 7.23 kB
/** * Copyright (c) 2015-present, Facebook, Inc. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. * * @flow strict */ import inspect from '../jsutils/inspect'; import isFinite from '../jsutils/isFinite'; import isInteger from '../jsutils/isInteger'; import { GraphQLScalarType, isNamedType } from './definition'; import { Kind } from '../language/kinds'; // As per the GraphQL Spec, Integers are only treated as valid when a valid // 32-bit signed integer, providing the broadest support across platforms. // // n.b. JavaScript's integers are safe between -(2^53 - 1) and 2^53 - 1 because // they are internally represented as IEEE 754 doubles. const MAX_INT = 2147483647; const MIN_INT = -2147483648; function serializeInt(value: mixed): number { if (typeof value === 'boolean') { return value ? 1 : 0; } let num = value; if (typeof value === 'string' && value !== '') { num = Number(value); } if (!isInteger(num)) { throw new TypeError( `Int cannot represent non-integer value: ${inspect(value)}`, ); } if (num > MAX_INT || num < MIN_INT) { throw new TypeError( `Int cannot represent non 32-bit signed integer value: ${inspect(value)}`, ); } return num; } function coerceInt(value: mixed): number { if (!isInteger(value)) { throw new TypeError( `Int cannot represent non-integer value: ${inspect(value)}`, ); } if (value > MAX_INT || value < MIN_INT) { throw new TypeError( `Int cannot represent non 32-bit signed integer value: ${inspect(value)}`, ); } return value; } export const GraphQLInt = new GraphQLScalarType({ name: 'Int', description: 'The `Int` scalar type represents non-fractional signed whole numeric ' + 'values. Int can represent values between -(2^31) and 2^31 - 1. ', serialize: serializeInt, parseValue: coerceInt, parseLiteral(ast) { if (ast.kind === Kind.INT) { const num = parseInt(ast.value, 10); if (num <= MAX_INT && num >= MIN_INT) { return num; } } return undefined; }, }); function serializeFloat(value: mixed): number { if (typeof value === 'boolean') { return value ? 1 : 0; } let num = value; if (typeof value === 'string' && value !== '') { num = Number(value); } if (!isFinite(num)) { throw new TypeError( `Float cannot represent non numeric value: ${inspect(value)}`, ); } return num; } function coerceFloat(value: mixed): number { if (!isFinite(value)) { throw new TypeError( `Float cannot represent non numeric value: ${inspect(value)}`, ); } return value; } export const GraphQLFloat = new GraphQLScalarType({ name: 'Float', description: 'The `Float` scalar type represents signed double-precision fractional ' + 'values as specified by ' + '[IEEE 754](http://en.wikipedia.org/wiki/IEEE_floating_point). ', serialize: serializeFloat, parseValue: coerceFloat, parseLiteral(ast) { return ast.kind === Kind.FLOAT || ast.kind === Kind.INT ? parseFloat(ast.value) : undefined; }, }); function serializeString(value: mixed): string { // Support serializing objects with custom valueOf() functions - a common way // to represent an complex value which can be represented as a string // (ex: MongoDB id objects). const result = value && typeof value.valueOf === 'function' ? value.valueOf() : value; // Serialize string, boolean and number values to a string, but do not // attempt to coerce object, function, symbol, or other types as strings. if (typeof result === 'string') { return result; } if (typeof result === 'boolean') { return result ? 'true' : 'false'; } if (isFinite(result)) { return result.toString(); } throw new TypeError(`String cannot represent value: ${inspect(value)}`); } function coerceString(value: mixed): string { if (typeof value !== 'string') { throw new TypeError( `String cannot represent a non string value: ${inspect(value)}`, ); } return value; } export const GraphQLString = new GraphQLScalarType({ name: 'String', description: 'The `String` scalar type represents textual data, represented as UTF-8 ' + 'character sequences. The String type is most often used by GraphQL to ' + 'represent free-form human-readable text.', serialize: serializeString, parseValue: coerceString, parseLiteral(ast) { return ast.kind === Kind.STRING ? ast.value : undefined; }, }); function serializeBoolean(value: mixed): boolean { if (typeof value === 'boolean') { return value; } if (isFinite(value)) { return value !== 0; } throw new TypeError( `Boolean cannot represent a non boolean value: ${inspect(value)}`, ); } function coerceBoolean(value: mixed): boolean { if (typeof value !== 'boolean') { throw new TypeError( `Boolean cannot represent a non boolean value: ${inspect(value)}`, ); } return value; } export const GraphQLBoolean = new GraphQLScalarType({ name: 'Boolean', description: 'The `Boolean` scalar type represents `true` or `false`.', serialize: serializeBoolean, parseValue: coerceBoolean, parseLiteral(ast) { return ast.kind === Kind.BOOLEAN ? ast.value : undefined; }, }); function serializeID(value: mixed): string { // Support serializing objects with custom valueOf() functions - a common way // to represent an object identifier (ex. MongoDB). const result = value && typeof value.valueOf === 'function' ? value.valueOf() : value; if (typeof result === 'string') { return result; } if (isInteger(result)) { return String(result); } throw new TypeError(`ID cannot represent value: ${inspect(value)}`); } function coerceID(value: mixed): string { if (typeof value === 'string') { return value; } if (isInteger(value)) { return value.toString(); } throw new TypeError(`ID cannot represent value: ${inspect(value)}`); } export const GraphQLID = new GraphQLScalarType({ name: 'ID', description: 'The `ID` scalar type represents a unique identifier, often used to ' + 'refetch an object or as key for a cache. The ID type appears in a JSON ' + 'response as a String; however, it is not intended to be human-readable. ' + 'When expected as an input type, any string (such as `"4"`) or integer ' + '(such as `4`) input value will be accepted as an ID.', serialize: serializeID, parseValue: coerceID, parseLiteral(ast) { return ast.kind === Kind.STRING || ast.kind === Kind.INT ? ast.value : undefined; }, }); export const specifiedScalarTypes: $ReadOnlyArray<*> = [ GraphQLString, GraphQLInt, GraphQLFloat, GraphQLBoolean, GraphQLID, ]; export function isSpecifiedScalarType(type: mixed): boolean %checks { return ( isNamedType(type) && // Would prefer to use specifiedScalarTypes.some(), however %checks needs // a simple expression. (type.name === GraphQLString.name || type.name === GraphQLInt.name || type.name === GraphQLFloat.name || type.name === GraphQLBoolean.name || type.name === GraphQLID.name) ); }