forked from microsoft/typespec
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Implement
@Interface
and @compose
decorators
The `@compose` decorator is used to indicate that a GraphQL `type` or `interface` implements one or more interfaces. As [defined by the GraphQL spec](https://spec.graphql.org/October2021/#sec-Interfaces), a `type` or `interface` implementing an interface must contain all the properties defined by that interface, and so we require that the TypeSpec model implement the interface's properties as well. There is no restriction on how this is accomplished (via spread, via composition, via manual copying of properties, etc). To reduce confusion, we do not allow a TypeSpec model to define both a `type` and an `interface`. In order to define an `interface`, the model must be decorated with the `@Interface` decorator.
- Loading branch information
Showing
7 changed files
with
456 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
import "../dist/src/lib/interface.js"; | ||
|
||
using TypeSpec.Reflection; | ||
|
||
namespace TypeSpec.GraphQL; | ||
|
||
/** | ||
* Mark this model as a GraphQL Interface. Interfaces can be implemented by other models. | ||
* | ||
* @example | ||
* | ||
* ```typespec | ||
* @Interface | ||
* model Person { | ||
* name: string; | ||
* } | ||
*/ | ||
extern dec Interface(target: Model); | ||
|
||
/** | ||
* Specify the GraphQL interfaces that should be implemented by a model. | ||
* The interfaces must be decorated with the @Interface decorator, | ||
* and all of the interfaces' properties must be present and compatible. | ||
* | ||
* @example | ||
* | ||
* ```typespec | ||
* @compose(Influencer, Person) | ||
* model User { | ||
* ... Influencer; | ||
* ... Person; | ||
* } | ||
*/ | ||
extern dec compose(target: Model, ...implements: Model[]); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1,2 @@ | ||
import "./interface.tsp"; | ||
import "./schema.tsp"; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,151 @@ | ||
import { | ||
type DecoratorContext, | ||
type DecoratorFunction, | ||
type Model, | ||
type ModelProperty, | ||
type Program, | ||
validateDecoratorTarget, | ||
validateDecoratorUniqueOnNode, | ||
walkPropertiesInherited, | ||
} from "@typespec/compiler"; | ||
|
||
import { GraphQLKeys, NAMESPACE, reportDiagnostic } from "../lib.js"; | ||
import type { Tagged } from "../types.d.ts"; | ||
import { useStateMap, useStateSet } from "./state-map.js"; | ||
|
||
// This will set the namespace for decorators implemented in this file | ||
export const namespace = NAMESPACE; | ||
|
||
/** An Interface is a model that has been marked as an Interface */ | ||
type Interface = Tagged<Model, "interface">; | ||
|
||
const [getInterface, setInterface] = useStateSet<Interface>(GraphQLKeys.interface); | ||
const [getComposition, setComposition, _getCompositionMap] = useStateMap<Model, Interface[]>( | ||
GraphQLKeys.compose, | ||
); | ||
|
||
export { | ||
/** | ||
* Get the implemented interfaces for a given model | ||
* @param program Program | ||
* @param model Model | ||
* @returns Composed interfaces or undefined if no interfaces are composed. | ||
*/ | ||
getComposition, | ||
}; | ||
|
||
/** | ||
* Check if the model is defined as a schema. | ||
* @param program Program | ||
* @param model Model | ||
* @returns Boolean | ||
*/ | ||
export function isInterface(program: Program, model: Model | Interface): model is Interface { | ||
return !!getInterface(program, model as Interface); | ||
} | ||
|
||
function validateImplementedsAreInterfaces(context: DecoratorContext, interfaces: Model[]) { | ||
let valid = true; | ||
|
||
for (const iface of interfaces) { | ||
if (!isInterface(context.program, iface)) { | ||
valid = false; | ||
reportDiagnostic(context.program, { | ||
code: "invalid-interface", | ||
format: { interface: iface.name }, | ||
target: context.decoratorTarget, | ||
}); | ||
} | ||
} | ||
|
||
return valid; | ||
} | ||
|
||
function validateNoCircularImplementation( | ||
context: DecoratorContext, | ||
target: Model, | ||
interfaces: Interface[], | ||
) { | ||
const valid = !isInterface(context.program, target) || !interfaces.includes(target); | ||
if (!valid) { | ||
reportDiagnostic(context.program, { | ||
code: "circular-interface", | ||
target: context.decoratorTarget, | ||
}); | ||
} | ||
return valid; | ||
} | ||
|
||
function propertiesEqual(prop1: ModelProperty, prop2: ModelProperty): boolean { | ||
// TODO is there some canonical way to do this? | ||
return ( | ||
prop1.name === prop2.name && prop1.type === prop2.type && prop1.optional === prop2.optional | ||
); | ||
} | ||
|
||
function validateImplementsInterfaceProperties( | ||
context: DecoratorContext, | ||
modelProperties: Map<string, ModelProperty>, | ||
iface: Interface, | ||
) { | ||
let valid = true; | ||
|
||
for (const prop of walkPropertiesInherited(iface)) { | ||
if (!modelProperties.has(prop.name)) { | ||
valid = false; | ||
reportDiagnostic(context.program, { | ||
code: "missing-interface-property", | ||
format: { interface: iface.name, property: prop.name }, | ||
target: context.decoratorTarget, | ||
}); | ||
} else if (!propertiesEqual(modelProperties.get(prop.name)!, prop)) { | ||
valid = false; | ||
reportDiagnostic(context.program, { | ||
code: "incompatible-interface-property", | ||
format: { interface: iface.name, property: prop.name }, | ||
target: context.decoratorTarget, | ||
}); | ||
} | ||
} | ||
|
||
return valid; | ||
} | ||
|
||
function validateImplementsInterfacesProperties( | ||
context: DecoratorContext, | ||
target: Model, | ||
interfaces: Interface[], | ||
) { | ||
let valid = true; | ||
const allModelProperties = new Map( | ||
[...walkPropertiesInherited(target)].map((prop) => [prop.name, prop]), | ||
); | ||
for (const iface of interfaces) { | ||
if (!validateImplementsInterfaceProperties(context, allModelProperties, iface)) { | ||
valid = false; | ||
} | ||
} | ||
return valid; | ||
} | ||
|
||
export const $Interface: DecoratorFunction = (context: DecoratorContext, target: Model) => { | ||
validateDecoratorTarget(context, target, "@Interface", "Model"); // TODO: Is this needed? https://github.com/Azure/cadl-azure/issues/1022 | ||
validateDecoratorUniqueOnNode(context, target, $Interface); | ||
setInterface(context.program, target as Interface); | ||
}; | ||
|
||
export const $compose: DecoratorFunction = ( | ||
context: DecoratorContext, | ||
target: Model, | ||
...interfaces: Interface[] | ||
) => { | ||
validateDecoratorTarget(context, target, "@compose", "Model"); // TODO: Is this needed? https://github.com/Azure/cadl-azure/issues/1022 | ||
validateImplementedsAreInterfaces(context, interfaces); | ||
validateNoCircularImplementation(context, target, interfaces); | ||
validateImplementsInterfacesProperties(context, target, interfaces); | ||
const existingCompose = getComposition(context.program, target); | ||
if (existingCompose) { | ||
interfaces = [...existingCompose, ...interfaces]; | ||
} | ||
setComposition(context.program, target, interfaces); | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,9 +1,12 @@ | ||
import type { DecoratorImplementations } from "@typespec/compiler"; | ||
import { NAMESPACE } from "./lib.js"; | ||
import { $compose, $Interface } from "./lib/interface.js"; | ||
import { $schema } from "./lib/schema.js"; | ||
|
||
export const $decorators: DecoratorImplementations = { | ||
[NAMESPACE]: { | ||
compose: $compose, | ||
Interface: $Interface, | ||
schema: $schema, | ||
}, | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.