UNPKG

dune-ts

Version:

Unofficial Dune Analytics Typescript Client

140 lines 15.7 kB
import { __decorate, __metadata } from "tslib"; import { EXECUTE_QUERY_BODY, HEADERS, QUERY_BODY, URLS } from './constants'; import { Cookies } from './Cookies'; import { isCookiesPresent, isCsrfPresent, isTokenPresent, maybeLogin, } from './decorators'; import { Parameters } from './Parameters'; import { delay } from './utils'; export class Dune { password; username; cookies; csrf; token; executionId; loggedAt; constructor(credentials = {}) { let { password, username } = credentials; password ??= process.env.DUNE_PASSWORD; username ??= process.env.DUNE_USERNAME; Object.entries(credentials).forEach(([key, value]) => { if (typeof value !== 'string') throw new Error(`${key} should be a string`); if (value === '') throw new Error(`${key} should be a non empty string`); }); this.password = password; this.username = username; this.cookies = new Cookies(); } async getCsrfToken() { const response = await fetch(URLS.CSRF, { method: 'POST' }); this.csrf = (await response.json()).csrf; this.cookies.set(response); } async getAuthCookies() { await fetch(URLS.AUTH, { body: new URLSearchParams({ action: 'login', csrf: this.csrf, next: URLS.BASE, password: this.password, username: this.username, }), headers: { ...HEADERS, 'Content-Type': 'application/x-www-form-urlencoded', cookie: `csrf=${this.csrf}`, }, method: 'POST', redirect: 'manual', }).then((res) => { this.cookies.set(res); }); } async getAuthToken() { const response = await fetch(URLS.SESSION, { body: 'false', headers: { ...HEADERS, cookie: this.cookies.toString(), }, method: 'POST', }); this.token = (await response.json()).token; } async login() { await this.getCsrfToken(); await this.getAuthCookies(); await this.getAuthToken(); this.loggedAt = new Date(); } async executeQuery(queryId, parameters = []) { const res = await fetch(URLS.GRAPH_EXEC_ID, { body: JSON.stringify({ ...EXECUTE_QUERY_BODY, variables: { parameters, query_id: queryId }, }), headers: { ...HEADERS, authorization: `Bearer ${this.token}`, 'Content-Type': 'application/json', }, method: 'POST', }); this.executionId = (await res.json()).data.execute_query_v2.job_id; } async query(queryId, parameterDatas) { const parameters = Parameters.create(parameterDatas); await this.executeQuery(queryId, parameters); let executionSucceeded = null; while (executionSucceeded === null) { const res = await fetch(URLS.GRAPH_QUERY, { body: JSON.stringify({ ...QUERY_BODY, variables: { execution_id: this.executionId, parameters, query_id: queryId, }, }), headers: { ...HEADERS, authorization: `Bearer ${this.token}`, 'Content-Type': 'application/json', }, method: 'POST', }); executionSucceeded = (await res.json()).data.get_execution .execution_succeeded; await delay(1500); } const { columns, data } = executionSucceeded; return { columns, data }; } } __decorate([ isCsrfPresent, __metadata("design:type", Function), __metadata("design:paramtypes", []), __metadata("design:returntype", Promise) ], Dune.prototype, "getAuthCookies", null); __decorate([ isCookiesPresent, __metadata("design:type", Function), __metadata("design:paramtypes", []), __metadata("design:returntype", Promise) ], Dune.prototype, "getAuthToken", null); __decorate([ isTokenPresent, __metadata("design:type", Function), __metadata("design:paramtypes", [Number, void 0]), __metadata("design:returntype", Promise) ], Dune.prototype, "executeQuery", null); __decorate([ maybeLogin(), __metadata("design:type", Function), __metadata("design:paramtypes", [Number, Array]), __metadata("design:returntype", Promise) ], Dune.prototype, "query", null); export const dune = new Dune(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Dune.js","sourceRoot":"src/","sources":["Dune.ts"],"names":[],"mappings":"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","sourcesContent":["import { EXECUTE_QUERY_BODY, HEADERS, QUERY_BODY, URLS } from './constants'\nimport { Cookies } from './Cookies'\nimport {\n  isCookiesPresent,\n  isCsrfPresent,\n  isTokenPresent,\n  maybeLogin,\n} from './decorators'\nimport { ParameterDatas, Parameters } from './Parameters'\nimport { delay } from './utils'\n\nexport class Dune {\n  private readonly password: string\n  private readonly username: string\n  private readonly cookies: Cookies\n  private csrf: string | undefined\n  private token: string | undefined\n  public executionId: string | undefined\n  private loggedAt: Date | undefined\n\n  constructor(credentials: { password?: string; username?: string } = {}) {\n    let { password, username } = credentials\n    password ??= process.env.DUNE_PASSWORD\n    username ??= process.env.DUNE_USERNAME\n\n    Object.entries(credentials).forEach(([key, value]) => {\n      if (typeof value !== 'string')\n        throw new Error(`${key} should be a string`)\n      if (value === '') throw new Error(`${key} should be a non empty string`)\n    })\n\n    // @ts-expect-error\n    this.password = password\n    // @ts-expect-error\n    this.username = username\n    this.cookies = new Cookies()\n  }\n\n  private async getCsrfToken() {\n    const response = await fetch(URLS.CSRF, { method: 'POST' })\n\n    this.csrf = (await response.json()).csrf\n    this.cookies.set(response)\n  }\n\n  @isCsrfPresent\n  private async getAuthCookies() {\n    await fetch(URLS.AUTH, {\n      // @ts-expect-error  - decorator already checks if csrf is defined\n      body: new URLSearchParams({\n        action: 'login',\n        csrf: this.csrf,\n        next: URLS.BASE,\n        password: this.password,\n        username: this.username,\n      }),\n      headers: {\n        ...HEADERS,\n        'Content-Type': 'application/x-www-form-urlencoded',\n        // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n        cookie: `csrf=${this.csrf}`,\n      },\n      method: 'POST',\n      redirect: 'manual',\n    }).then((res) => {\n      this.cookies.set(res)\n    })\n  }\n\n  @isCookiesPresent\n  private async getAuthToken() {\n    const response = await fetch(URLS.SESSION, {\n      body: 'false',\n      headers: {\n        ...HEADERS,\n        cookie: this.cookies.toString(),\n      },\n      method: 'POST',\n    })\n\n    this.token = (await response.json()).token\n  }\n\n  public async login() {\n    await this.getCsrfToken()\n    await this.getAuthCookies()\n    await this.getAuthToken()\n    this.loggedAt = new Date()\n  }\n\n  @isTokenPresent\n  private async executeQuery(\n    queryId: number,\n    parameters: ReturnType<typeof Parameters.create> = [],\n  ) {\n    const res = await fetch(URLS.GRAPH_EXEC_ID, {\n      body: JSON.stringify({\n        ...EXECUTE_QUERY_BODY,\n        variables: { parameters, query_id: queryId },\n      }),\n      headers: {\n        ...HEADERS,\n        // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n        authorization: `Bearer ${this.token}`,\n        'Content-Type': 'application/json',\n      },\n      method: 'POST',\n    })\n\n    this.executionId = (await res.json()).data.execute_query_v2.job_id\n  }\n\n  @maybeLogin()\n  public async query(queryId: number, parameterDatas?: ParameterDatas) {\n    const parameters = Parameters.create(parameterDatas)\n    await this.executeQuery(queryId, parameters)\n\n    let executionSucceeded: null | { columns: string[]; data: any[] } = null\n\n    while (executionSucceeded === null) {\n      const res = await fetch(URLS.GRAPH_QUERY, {\n        body: JSON.stringify({\n          ...QUERY_BODY,\n          variables: {\n            execution_id: this.executionId,\n            parameters,\n            query_id: queryId,\n          },\n        }),\n        headers: {\n          ...HEADERS,\n          // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n          authorization: `Bearer ${this.token}`,\n          'Content-Type': 'application/json',\n        },\n        method: 'POST',\n      })\n      executionSucceeded = (await res.json()).data.get_execution\n        .execution_succeeded\n\n      await delay(1500)\n    }\n\n    const { columns, data } = executionSucceeded\n    return { columns, data }\n  }\n}\n\nexport const dune = new Dune()\n"]}