Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat!: accept DocumentNode input #183

Merged
merged 5 commits into from
Aug 3, 2020
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,8 @@
"dripip": "^0.9.0",
"express": "^4.17.1",
"fetch-cookie": "0.7.2",
"graphql": "^15.3.0",
jasonkuhrt marked this conversation as resolved.
Show resolved Hide resolved
"graphql-tag": "^2.11.0",
"jest": "^26.0.1",
"prettier": "^2.0.5",
"ts-jest": "^26.0.0",
Expand Down
98 changes: 89 additions & 9 deletions src/index.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,13 @@
import fetch from 'cross-fetch'
import { ClientError, GraphQLError, Variables } from './types'
import { print } from 'graphql/language/printer'
import { ClientError, GraphQLError, RequestDocument, Variables } from './types'
import { RequestInit, Response } from './types.dom'

export { ClientError } from './types'

/**
* todo
*/
export class GraphQLClient {
private url: string
private options: RequestInit
Expand Down Expand Up @@ -45,11 +49,15 @@ export class GraphQLClient {
}
}

async request<T = any, V = Variables>(query: string, variables?: V): Promise<T> {
/**
* todo
*/
async request<T = any, V = Variables>(document: RequestDocument, variables?: V): Promise<T> {
const { headers, ...others } = this.options
const resolvedDoc = resolveRequestDocument(document)

const body = JSON.stringify({
query,
query: resolvedDoc,
variables: variables ? variables : undefined,
})

Expand All @@ -66,13 +74,12 @@ export class GraphQLClient {
return result.data
} else {
const errorResult = typeof result === 'string' ? { error: result } : result
throw new ClientError({ ...errorResult, status: response.status }, { query, variables })
throw new ClientError({ ...errorResult, status: response.status }, { query: resolvedDoc, variables })
}
}

setHeaders(headers: Response['headers']): GraphQLClient {
this.options.headers = headers

return this
}

Expand All @@ -86,28 +93,71 @@ export class GraphQLClient {
} else {
this.options.headers = { [key]: value }
}

return this
}
}

/**
* todo
*/
export async function rawRequest<T = any, V = Variables>(
url: string,
query: string,
variables?: V
): Promise<{ data?: T; extensions?: any; headers: Headers; status: number; errors?: GraphQLError[] }> {
const client = new GraphQLClient(url)

return client.rawRequest<T, V>(query, variables)
}

export async function request<T = any, V = Variables>(url: string, query: string, variables?: V): Promise<T> {
/**
* Send a GraphQL Document to the GraphQL server for exectuion.
*
* @example
*
* ```ts
* // You can pass a raw string
*
* await request('https://foo.bar/graphql', `
* {
* query {
* users
* }
* }
* `)
*
* // You can also pass a GraphQL DocumentNode. Convenient if you
* // are using graphql-tag package.
*
* import gql from 'graphql-tag'
*
* await request('https://foo.bar/graphql', gql`...`)
*
* // If you don't actually care about using DocumentNode but just
* // want the tooling support for gql template tag like IDE syntax
* // coloring and prettier autoformat then note you can use the
* // passthrough gql tag shipped with graphql-request to save a bit
* // of performance and not have to install another dep into your project.
*
* import { gql } from 'graphql-request'
*
* await request('https://foo.bar/graphql', gql`...`)
* ```
*/
export async function request<T = any, V = Variables>(
url: string,
document: RequestDocument,
variables?: V
): Promise<T> {
const client = new GraphQLClient(url)

return client.request<T, V>(query, variables)
return client.request<T, V>(document, variables)
}

export default request

/**
* todo
*/
function getResult(response: Response): Promise<any> {
const contentType = response.headers.get('Content-Type')
if (contentType && contentType.startsWith('application/json')) {
Expand All @@ -116,3 +166,33 @@ function getResult(response: Response): Promise<any> {
return response.text()
}
}

/**
* helpers
*/

function resolveRequestDocument(document: RequestDocument): string {
if (typeof document === 'string') return document

return print(document)
}

/**
* Convenience passthrough template tag to get the benefits of tooling for the gql template tag. This does not actually parse the input into a GraphQL DocumentNode like graphql-tag package does. It just returns the string with any variables given interpolated. Can save you a bit of performance and having to install another package.
*
* @example
*
* import { gql } from 'graphql-request'
*
* await request('https://foo.bar/graphql', gql`...`)
*
* @remarks
*
* Several tools in the Node GraphQL ecosystem are hardcoded to specially treat any template tag named "gql". For example see this prettier issue: https://github.com/prettier/prettier/issues/4360. Using this template tag has no runtime effect beyond variable interpolation.
*/
export function gql(chunks: TemplateStringsArray, ...variables: any[]): string {
return chunks.reduce(
(accumulator, chunk, index) => `${accumulator}${chunk}${index in variables ? variables[index] : ''}`,
''
)
}
4 changes: 4 additions & 0 deletions src/types.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import { DocumentNode } from 'graphql/language/ast'

export type Variables = { [key: string]: any }

export interface GraphQLError {
Expand Down Expand Up @@ -50,3 +52,5 @@ export class ClientError extends Error {
}
}
}

export type RequestDocument = string | DocumentNode
24 changes: 21 additions & 3 deletions tests/__helpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,25 @@ type Context = {
server: Application
nodeServer: Server
url: string
mock: <D extends JsonObject>(data: D) => D & { requests: CapturedRequest[] }
res: <S extends MockSpec>(spec: S) => MockResult<S>
}

type MockSpec = {
headers?: Record<string, string>
body?: {
data?: JsonObject
extensions?: JsonObject
errors?: JsonObject
}
}

type MockResult<Spec extends MockSpec> = {
spec: Spec
requests: {
method: string
headers: Record<string, string>
body: JsonObject
}[]
}

export function setupTestServer() {
Expand All @@ -22,7 +40,7 @@ export function setupTestServer() {
ctx.url = 'http://localhost:3210'
ctx.nodeServer.on('request', ctx.server)
ctx.nodeServer.once('listening', done)
ctx.mock = (spec) => {
ctx.res = (spec) => {
const requests: CapturedRequest[] = []
ctx.server.use('*', function mock(req, res) {
requests.push({
Expand All @@ -37,7 +55,7 @@ export function setupTestServer() {
}
res.send(spec.body ?? {})
})
return { ...spec, requests }
return { spec, requests: requests as any } as any
}
})

Expand Down
80 changes: 67 additions & 13 deletions tests/index.test.ts
Original file line number Diff line number Diff line change
@@ -1,24 +1,25 @@
import { GraphQLClient, rawRequest, request } from '../src'
import graphqlTag from 'graphql-tag'
import { gql, GraphQLClient, rawRequest, request } from '../src'
import { setupTestServer } from './__helpers'

const ctx = setupTestServer()

test('minimal query', async () => {
const data = ctx.mock({
const { data } = ctx.res({
body: {
data: {
viewer: {
id: 'some-id',
},
},
},
}).body.data
}).spec.body

expect(await request(ctx.url, `{ viewer { id } }`)).toEqual(data)
})

test('minimal raw query', async () => {
const { extensions, data } = ctx.mock({
const { extensions, data } = ctx.res({
body: {
data: {
viewer: {
Expand All @@ -29,7 +30,7 @@ test('minimal raw query', async () => {
version: '1',
},
},
}).body
}).spec.body
const { headers, ...result } = await rawRequest(ctx.url, `{ viewer { id } }`)
expect(result).toEqual({ data, extensions, status: 200 })
})
Expand All @@ -38,7 +39,7 @@ test('minimal raw query with response headers', async () => {
const {
headers: reqHeaders,
body: { data, extensions },
} = ctx.mock({
} = ctx.res({
headers: {
'Content-Type': 'application/json',
'X-Custom-Header': 'test-custom-header',
Expand All @@ -53,15 +54,15 @@ test('minimal raw query with response headers', async () => {
version: '1',
},
},
})
}).spec
const { headers, ...result } = await rawRequest(ctx.url, `{ viewer { id } }`)

expect(result).toEqual({ data, extensions, status: 200 })
expect(headers.get('X-Custom-Header')).toEqual(reqHeaders['X-Custom-Header'])
})

test('content-type with charset', async () => {
const { data } = ctx.mock({
const { data } = ctx.res({
// headers: { 'Content-Type': 'application/json; charset=utf-8' },
body: {
data: {
Expand All @@ -70,13 +71,13 @@ test('content-type with charset', async () => {
},
},
},
}).body
}).spec.body

expect(await request(ctx.url, `{ viewer { id } }`)).toEqual(data)
})

test('basic error', async () => {
ctx.mock({
ctx.res({
body: {
errors: {
message: 'Syntax Error GraphQL request (1:1) Unexpected Name "x"\n\n1: x\n ^\n',
Expand All @@ -88,7 +89,7 @@ test('basic error', async () => {
],
},
},
}).body
})

const res = await request(ctx.url, `x`).catch((x) => x)

Expand All @@ -98,7 +99,7 @@ test('basic error', async () => {
})

test('basic error with raw request', async () => {
ctx.mock({
ctx.res({
body: {
errors: {
message: 'Syntax Error GraphQL request (1:1) Unexpected Name "x"\n\n1: x\n ^\n',
Expand Down Expand Up @@ -127,7 +128,7 @@ test.skip('extra fetch options', async () => {
}

const client = new GraphQLClient(ctx.url, options)
const { requests } = ctx.mock({
const { requests } = ctx.res({
body: { data: { test: 'test' } },
})
await client.request('{ test }')
Expand All @@ -151,3 +152,56 @@ test.skip('extra fetch options', async () => {
]
`)
})

describe('DocumentNode', () => {
it('accepts graphql DocumentNode as alternative to raw string', async () => {
const mock = ctx.res({ body: { data: { foo: 1 } } })
await request(
ctx.url,
graphqlTag`
{
query {
users
}
}
`
)
expect(mock.requests[0].body).toMatchInlineSnapshot(`
Object {
"query": "{
query {
users
}
}
",
}
`)
})
})

describe('gql', () => {
it('passthrough allowing benefits of tooling for gql template tag', async () => {
const mock = ctx.res({ body: { data: { foo: 1 } } })
await request(
ctx.url,
gql`
{
query {
users
}
}
`
)
expect(mock.requests[0].body).toMatchInlineSnapshot(`
Object {
"query": "
{
query {
users
}
}
",
}
`)
})
})
Loading