-
-
Notifications
You must be signed in to change notification settings - Fork 349
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
8 changed files
with
279 additions
and
56 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
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 @@ | ||
export class ConfigurationError extends Error {} |
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,210 @@ | ||
import { gql } from 'graphql-tag'; | ||
import { ASTNode, print } from 'graphql'; | ||
import { isUndefined } from 'lodash'; | ||
import { reject } from 'ramda'; | ||
|
||
import { ConfigurationError } from './configurationError'; | ||
import { GraphQLQueryError } from './graphQLQueryError'; | ||
import { PactV3 } from '../pact'; | ||
import { GraphQLVariables } from '../../dsl/graphql'; | ||
import { V3Request, V3Response } from '../types'; | ||
import { OperationType } from './types'; | ||
import { JsonMap } from '../../common/jsonTypes'; | ||
|
||
import { regex } from '../matchers'; | ||
|
||
const escapeSpace = (s: string) => s.replace(/\s+/g, '\\s*'); | ||
|
||
const escapeRegexChars = (s: string) => | ||
s.replace(/[-[\]/{}()*+?.\\^$|]/g, '\\$&'); | ||
|
||
const escapeGraphQlQuery = (s: string) => escapeSpace(escapeRegexChars(s)); | ||
|
||
/** | ||
* Accepts a raw or pre-parsed query, validating in the former case, and | ||
* returns a normalized raw query. | ||
* @param query {string|ASTNode} the query to validate | ||
* @param type the operation type | ||
*/ | ||
function validateQuery(query: string | ASTNode, type: OperationType): string { | ||
if (!query) { | ||
throw new ConfigurationError(`You must provide a GraphQL ${type}.`); | ||
} | ||
|
||
if (typeof query !== 'string') { | ||
if (query?.kind === 'Document') { | ||
// Already parsed, store in string form | ||
return print(query); | ||
} | ||
throw new ConfigurationError( | ||
'You must provide a either a string or parsed GraphQL.' | ||
); | ||
} else { | ||
// String, so validate it | ||
try { | ||
gql(query); | ||
} catch (e) { | ||
throw new GraphQLQueryError(`GraphQL ${type} is invalid: ${e.message}`); | ||
} | ||
|
||
return query; | ||
} | ||
} | ||
|
||
/** | ||
* Expose a V3 compatible GraphQL interface | ||
* | ||
* Code borrowed/inspired from https://gist.github.com/wabrit/2d1e1f9520aa133908f0a3716338e5ff | ||
*/ | ||
export class GraphQLPactV3 extends PactV3 { | ||
private operation?: string = undefined; | ||
|
||
private variables?: GraphQLVariables = undefined; | ||
|
||
private query: string; | ||
|
||
private req?: V3Request = undefined; | ||
|
||
public given(providerState: string, parameters?: JsonMap): GraphQLPactV3 { | ||
super.given(providerState, parameters); | ||
|
||
return this; | ||
} | ||
|
||
public uponReceiving(description: string): GraphQLPactV3 { | ||
super.uponReceiving(description); | ||
|
||
return this; | ||
} | ||
|
||
/** | ||
* The GraphQL operation name, if used. | ||
* @param operation {string} the name of the operation | ||
* @return this object | ||
*/ | ||
withOperation(operation: string): GraphQLPactV3 { | ||
this.operation = operation; | ||
return this; | ||
} | ||
|
||
/** | ||
* Add variables used in the Query. | ||
* @param variables {GraphQLVariables} | ||
* @return this object | ||
*/ | ||
withVariables(variables: GraphQLVariables): GraphQLPactV3 { | ||
this.variables = variables; | ||
return this; | ||
} | ||
|
||
/** | ||
* The actual GraphQL query as a string. | ||
* | ||
* NOTE: spaces are not important, Pact will auto-generate a space-insensitive matcher | ||
* | ||
* e.g. the value for the "query" field in the GraphQL HTTP payload: | ||
* '{ "query": "{ | ||
* Category(id:7) { | ||
* id, | ||
* name, | ||
* subcategories { | ||
* id, | ||
* name | ||
* } | ||
* } | ||
* }" | ||
* }' | ||
* @param query {string|ASTNode} parsed or unparsed query | ||
* @return this object | ||
*/ | ||
withQuery(query: string | ASTNode): GraphQLPactV3 { | ||
this.query = validateQuery(query, OperationType.Query); | ||
|
||
return this; | ||
} | ||
|
||
/** | ||
* The actual GraphQL mutation as a string or parse tree. | ||
* | ||
* NOTE: spaces are not important, Pact will auto-generate a space-insensitive matcher | ||
* | ||
* e.g. the value for the "query" field in the GraphQL HTTP payload: | ||
* | ||
* mutation CreateReviewForEpisode($ep: Episode!, $review: ReviewInput!) { | ||
* createReview(episode: $ep, review: $review) { | ||
* stars | ||
* commentary | ||
* } | ||
* } | ||
* @param mutation {string|ASTNode} parsed or unparsed mutation | ||
* @return this object | ||
*/ | ||
withMutation(mutation: string | ASTNode): GraphQLPactV3 { | ||
this.query = validateQuery(mutation, OperationType.Mutation); | ||
|
||
return this; | ||
} | ||
|
||
/** | ||
* Used to pass in the method, path and content-type; the body detail would | ||
* not typically be passed here as that will be internally constructed from | ||
* withQuery/withMutation/withVariables calls. | ||
* | ||
* @see {@link withQuery} | ||
* @see {@link withMutation} | ||
* @see {@link withVariables} | ||
* @param req {V3Request} request | ||
* @return this object | ||
*/ | ||
withRequest(req: V3Request): GraphQLPactV3 { | ||
// Just take what we need from the request, as most of the detail will | ||
// come from withQuery/withMutation/withVariables | ||
this.req = req; | ||
return this; | ||
} | ||
|
||
/** | ||
* Overridden as this is the "trigger point" by which we should have received all | ||
* request information. | ||
* @param res {V3Response} the expected response | ||
* @returns this object | ||
*/ | ||
willRespondWith(res: V3Response): GraphQLPactV3 { | ||
if (!this.query) { | ||
throw new ConfigurationError('You must provide a GraphQL query.'); | ||
} | ||
|
||
if (!this.req) { | ||
throw new ConfigurationError('You must provide a GraphQL request.'); | ||
} | ||
|
||
this.req = { | ||
...this.req, | ||
body: reject(isUndefined, { | ||
operationName: this.operation, | ||
query: regex(escapeGraphQlQuery(this.query), this.query), | ||
variables: this.variables, | ||
}), | ||
headers: { | ||
'Content-Type': (this.req.contentType ||= 'application/json'), | ||
}, | ||
method: (this.req.method ||= 'POST'), | ||
}; | ||
|
||
super.withRequest(this.req); | ||
super.willRespondWith(res); | ||
return this; | ||
} | ||
|
||
public addInteraction(): GraphQLPactV3 { | ||
throw new ConfigurationError('Only GraphQL Queries are allowed'); | ||
} | ||
|
||
public withRequestBinaryFile(): PactV3 { | ||
throw new ConfigurationError('Only GraphQL Queries are allowed'); | ||
} | ||
|
||
public withRequestMultipartFileUpload(): PactV3 { | ||
throw new ConfigurationError('Only GraphQL Queries are allowed'); | ||
} | ||
} |
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 @@ | ||
export class GraphQLQueryError extends Error {} |
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,4 @@ | ||
export * from './graphQL'; | ||
export * from './configurationError'; | ||
export * from './graphQLQueryError'; | ||
export * from './types'; |
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,4 @@ | ||
export enum OperationType { | ||
Mutation = 'Mutation', | ||
Query = 'Query', | ||
} |
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