Skip to content

Commit

Permalink
ease upgrade path for programmatic default values
Browse files Browse the repository at this point in the history
  • Loading branch information
yaacovCR committed Nov 18, 2024
1 parent 1d98a6a commit 4b489ab
Show file tree
Hide file tree
Showing 28 changed files with 321 additions and 222 deletions.
2 changes: 1 addition & 1 deletion integrationTests/ts/basic-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ const queryType: GraphQLObjectType = new GraphQLObjectType({
args: {
who: {
type: GraphQLString,
defaultValue: 'World',
externalDefaultValue: 'World',
},
},
resolve(_root, args: { who: string }) {
Expand Down
2 changes: 1 addition & 1 deletion integrationTests/ts/esm.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ const queryType: GraphQLObjectType = new GraphQLObjectType({
args: {
who: {
type: GraphQLString,
defaultValue: 'World',
externalDefaultValue: 'World',
},
},
resolve(_root, args: { who: string }) {
Expand Down
2 changes: 1 addition & 1 deletion src/execution/__tests__/executor-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -246,7 +246,7 @@ describe('Execute: Handles basic execution tasks', () => {
signature: {
name: 'var',
type: GraphQLString,
defaultValue: undefined,
externalDefaultValue: undefined,
},
value: 'abc',
},
Expand Down
6 changes: 3 additions & 3 deletions src/execution/__tests__/variables-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -142,11 +142,11 @@ const TestType = new GraphQLObjectType({
}),
fieldWithDefaultArgumentValue: fieldWithInputArg({
type: GraphQLString,
defaultValue: 'Hello World',
externalDefaultValue: 'Hello World',
}),
fieldWithNonNullableStringInputAndDefaultArgumentValue: fieldWithInputArg({
type: new GraphQLNonNull(GraphQLString),
defaultValue: 'Hello World',
externalDefaultValue: 'Hello World',
}),
fieldWithNestedInputObject: fieldWithInputArg({
type: TestNestedInputObject,
Expand Down Expand Up @@ -187,7 +187,7 @@ const schema = new GraphQLSchema({
type: new GraphQLNonNull(GraphQLBoolean),
description: 'Skipped when true.',
// default values will override operation variables in the setting of defined fragment variables that are not provided
defaultValue: true,
externalDefaultValue: true,
},
},
}),
Expand Down
16 changes: 8 additions & 8 deletions src/execution/getVariableSignature.ts
Original file line number Diff line number Diff line change
@@ -1,14 +1,13 @@
import { GraphQLError } from '../error/GraphQLError.js';

import type { VariableDefinitionNode } from '../language/ast.js';
import type {
ConstValueNode,
VariableDefinitionNode,
} from '../language/ast.js';
import { print } from '../language/printer.js';

import { isInputType } from '../type/definition.js';
import type {
GraphQLDefaultValueUsage,
GraphQLInputType,
GraphQLSchema,
} from '../type/index.js';
import type { GraphQLInputType, GraphQLSchema } from '../type/index.js';

import { typeFromAST } from '../utilities/typeFromAST.js';

Expand All @@ -21,7 +20,8 @@ import { typeFromAST } from '../utilities/typeFromAST.js';
export interface GraphQLVariableSignature {
name: string;
type: GraphQLInputType;
defaultValue: GraphQLDefaultValueUsage | undefined;
defaultValue?: never;
externalDefaultValue: { literal: ConstValueNode } | undefined;
}

export function getVariableSignature(
Expand All @@ -46,6 +46,6 @@ export function getVariableSignature(
return {
name: varName,
type: varType,
defaultValue: defaultValue ? { literal: defaultValue } : undefined,
externalDefaultValue: defaultValue ? { literal: defaultValue } : undefined,
};
}
16 changes: 6 additions & 10 deletions src/execution/values.ts
Original file line number Diff line number Diff line change
Expand Up @@ -226,11 +226,9 @@ export function experimentalGetArgumentValues(
{ nodes: node },
);
}
if (argDef.defaultValue) {
coercedValues[name] = coerceDefaultValue(
argDef.defaultValue,
argDef.type,
);
const coercedDefaultValue = coerceDefaultValue(argDef);
if (coercedDefaultValue !== undefined) {
coercedValues[name] = coercedDefaultValue;
}
continue;
}
Expand All @@ -249,11 +247,9 @@ export function experimentalGetArgumentValues(
!Object.hasOwn(scopedVariableValues.coerced, variableName)) &&
!isRequiredArgument(argDef)
) {
if (argDef.defaultValue) {
coercedValues[name] = coerceDefaultValue(
argDef.defaultValue,
argDef.type,
);
const coercedDefaultValue = coerceDefaultValue(argDef);
if (coercedDefaultValue !== undefined) {
coercedValues[name] = coercedDefaultValue;
}
continue;
}
Expand Down
22 changes: 15 additions & 7 deletions src/type/__tests__/definition-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -195,7 +195,8 @@ describe('Type System: Objects', () => {
input: {
description: 'Argument description.',
type: ScalarType,
defaultValue: 'DefaultValue',
defaultValue: undefined,
externalDefaultValue: 'DefaultValue',
defaultValueLiteral: undefined,
deprecationReason: 'Argument deprecation reason.',
extensions: { someExtension: 'extension' },
Expand Down Expand Up @@ -343,6 +344,7 @@ describe('Type System: Objects', () => {
description: undefined,
type: ScalarType,
defaultValue: undefined,
externalDefaultValue: undefined,
deprecationReason: undefined,
extensions: {},
astNode: undefined,
Expand Down Expand Up @@ -467,6 +469,7 @@ describe('Type System: Interfaces', () => {
description: 'Argument description.',
type: ScalarType,
defaultValue: undefined,
externalDefaultValue: undefined,
defaultValueLiteral: dummyAny,
deprecationReason: 'Argument deprecation reason.',
extensions: { someExtension: 'extension' },
Expand Down Expand Up @@ -802,7 +805,8 @@ describe('Type System: Input Objects', () => {
input: {
description: 'Argument description.',
type: ScalarType,
defaultValue: 'DefaultValue',
defaultValue: undefined,
externalDefaultValue: 'DefaultValue',
defaultValueLiteral: undefined,
deprecationReason: 'Argument deprecation reason.',
extensions: { someExtension: 'extension' },
Expand Down Expand Up @@ -832,6 +836,7 @@ describe('Type System: Input Objects', () => {
description: undefined,
type: ScalarType,
defaultValue: undefined,
externalDefaultValue: undefined,
deprecationReason: undefined,
extensions: {},
astNode: undefined,
Expand All @@ -852,6 +857,7 @@ describe('Type System: Input Objects', () => {
description: undefined,
type: ScalarType,
defaultValue: undefined,
externalDefaultValue: undefined,
extensions: {},
deprecationReason: undefined,
astNode: undefined,
Expand Down Expand Up @@ -901,14 +907,15 @@ describe('Type System: Input Objects', () => {
const inputObjType = new GraphQLInputObjectType({
name: 'SomeInputObject',
fields: {
f: { type: ScalarType, defaultValue: 3 },
f: { type: ScalarType, externalDefaultValue: 3 },
},
});
expect(inputObjType.getFields().f).to.deep.include({
name: 'f',
description: undefined,
type: ScalarType,
defaultValue: { value: 3 },
defaultValue: undefined,
externalDefaultValue: { value: 3 },
deprecationReason: undefined,
extensions: {},
astNode: undefined,
Expand All @@ -929,7 +936,8 @@ describe('Type System: Input Objects', () => {
name: 'f',
description: undefined,
type: ScalarType,
defaultValue: { literal: { kind: 'IntValue', value: '3' } },
defaultValue: undefined,
externalDefaultValue: { literal: { kind: 'IntValue', value: '3' } },
deprecationReason: undefined,
extensions: {},
astNode: undefined,
Expand All @@ -942,13 +950,13 @@ describe('Type System: Input Objects', () => {
fields: {
f: {
type: ScalarType,
defaultValue: 3,
externalDefaultValue: 3,
defaultValueLiteral: { kind: Kind.INT, value: '3' },
},
},
});
expect(() => inputObjType.getFields()).to.throw(
'Argument "f" has both a defaultValue and a defaultValueLiteral property, but only one must be provided.',
'Argument "f" has both an externalDefaultValue and a defaultValueLiteral property, but only one must be provided.',
);
});
});
Expand Down
2 changes: 2 additions & 0 deletions src/type/__tests__/directive-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ describe('Type System: Directive', () => {
description: undefined,
type: GraphQLString,
defaultValue: undefined,
externalDefaultValue: undefined,
deprecationReason: undefined,
extensions: {},
astNode: undefined,
Expand All @@ -48,6 +49,7 @@ describe('Type System: Directive', () => {
description: undefined,
type: GraphQLInt,
defaultValue: undefined,
externalDefaultValue: undefined,
deprecationReason: undefined,
extensions: {},
astNode: undefined,
Expand Down
28 changes: 27 additions & 1 deletion src/type/__tests__/enumType-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,7 @@ const QueryType = new GraphQLObjectType({
args: {
fromEnum: {
type: ComplexEnum,
defaultValue: 'ONE',
externalDefaultValue: 'ONE',
},
provideGoodValue: { type: GraphQLBoolean },
provideBadValue: { type: GraphQLBoolean },
Expand All @@ -90,6 +90,18 @@ const QueryType = new GraphQLObjectType({
return fromEnum;
},
},
complexEnumWithLegacyDefault: {
type: ComplexEnum,
args: {
fromEnum: {
type: ComplexEnum,
defaultValue: Complex1,
},
},
resolve(_source, { fromEnum }) {
return fromEnum;
},
},
thunkValuesString: {
type: GraphQLString,
args: {
Expand Down Expand Up @@ -438,6 +450,20 @@ describe('Type System: Enum Values', () => {
});
});

it('may be internally represented with complex values using legacy internal defaults', () => {
const result = executeQuery(`
{
complexEnumWithLegacyDefault
}
`);

expectJSON(result).toDeepEqual({
data: {
complexEnumWithLegacyDefault: 'ONE',
},
});
});

it('may have values specified via a callback', () => {
const result = executeQuery('{ thunkValuesString(fromEnum: B) }');

Expand Down
26 changes: 14 additions & 12 deletions src/type/__tests__/predicate-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -568,15 +568,16 @@ describe('Type predicates', () => {
describe('isRequiredArgument', () => {
function buildArg(config: {
type: GraphQLInputType;
defaultValue?: unknown;
externalDefaultValue?: unknown;
}): GraphQLArgument {
return {
name: 'someArg',
type: config.type,
description: undefined,
defaultValue:
config.defaultValue !== undefined
? { value: config.defaultValue }
defaultValue: undefined,
externalDefaultValue:
config.externalDefaultValue !== undefined
? { value: config.externalDefaultValue }
: undefined,
deprecationReason: null,
extensions: Object.create(null),
Expand All @@ -599,7 +600,7 @@ describe('Type predicates', () => {

const optArg2 = buildArg({
type: GraphQLString,
defaultValue: null,
externalDefaultValue: null,
});
expect(isRequiredArgument(optArg2)).to.equal(false);

Expand All @@ -610,7 +611,7 @@ describe('Type predicates', () => {

const optArg4 = buildArg({
type: new GraphQLNonNull(GraphQLString),
defaultValue: 'default',
externalDefaultValue: 'default',
});
expect(isRequiredArgument(optArg4)).to.equal(false);
});
Expand All @@ -619,15 +620,16 @@ describe('Type predicates', () => {
describe('isRequiredInputField', () => {
function buildInputField(config: {
type: GraphQLInputType;
defaultValue?: unknown;
externalDefaultValue?: unknown;
}): GraphQLInputField {
return {
name: 'someInputField',
type: config.type,
description: undefined,
defaultValue:
config.defaultValue !== undefined
? { value: config.defaultValue }
defaultValue: undefined,
externalDefaultValue:
config.externalDefaultValue !== undefined
? { value: config.externalDefaultValue }
: undefined,
deprecationReason: null,
extensions: Object.create(null),
Expand All @@ -650,7 +652,7 @@ describe('Type predicates', () => {

const optField2 = buildInputField({
type: GraphQLString,
defaultValue: null,
externalDefaultValue: null,
});
expect(isRequiredInputField(optField2)).to.equal(false);

Expand All @@ -661,7 +663,7 @@ describe('Type predicates', () => {

const optField4 = buildInputField({
type: new GraphQLNonNull(GraphQLString),
defaultValue: 'default',
externalDefaultValue: 'default',
});
expect(isRequiredInputField(optField4)).to.equal(false);
});
Expand Down
Loading

0 comments on commit 4b489ab

Please sign in to comment.