Skip to content
Snippets Groups Projects
graphql-client.ts 2.33 KiB
/**
 * Copyright (c) 2018 mol* contributors, licensed under MIT, See LICENSE file for more info.
 *
 * @author Alexander Rose <alexander.rose@weirdbyte.de>
 *
 * Adapted from https://github.com/prisma/graphql-request, Copyright (c) 2017 Graphcool, MIT
 */

import { RuntimeContext } from '../mol-task';

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

interface GraphQLError {
    message: string
    locations: { line: number, column: number }[]
    path: string[]
}

interface GraphQLResponse {
    data?: any
    errors?: GraphQLError[]
    extensions?: any
    status: number
    [key: string]: any
}

interface GraphQLRequestContext {
    query: string
    variables?: Variables
}

export class ClientError extends Error {
    response: GraphQLResponse
    request: GraphQLRequestContext

    constructor (response: GraphQLResponse, request: GraphQLRequestContext) {
        const message = `${ClientError.extractMessage(response)}: ${JSON.stringify({ response, request })}`

        super(message)

        this.response = response
        this.request = request

        // this is needed as Safari doesn't support .captureStackTrace
        /* tslint:disable-next-line */
        if (typeof Error.captureStackTrace === 'function') {
            Error.captureStackTrace(this, ClientError)
        }
    }

    private static extractMessage (response: GraphQLResponse): string {
        try {
            return response.errors![0].message
        } catch (e) {
            return `GraphQL Error (Code: ${response.status})`
        }
    }
}

export class GraphQLClient {
    constructor(private url: string, private fetch: import('../mol-util/data-source').AjaxTask) {
        this.url = url
    }

    async request<T extends any>(ctx: RuntimeContext, query: string, variables?: Variables): Promise<T> {

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

        const resultStr = await this.fetch({ url: this.url, body }).runInContext(ctx)
        const result = JSON.parse(resultStr)

        if (!result.errors && result.data) {
            return result.data
        } else {
            const errorResult = typeof result === 'string' ? { error: result } : result
            throw new ClientError(
                { ...errorResult },
                { query, variables },
            )
        }
    }
}