UNPKG

dune-ts

Version:

Unofficial Dune Analytics Typescript Client

128 lines 16.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.dune = exports.Dune = void 0; const tslib_1 = require("tslib"); const constants_1 = require("./constants"); const Cookies_1 = require("./Cookies"); const decorators_1 = require("./decorators"); const Parameters_1 = require("./Parameters"); const utils_1 = require("./utils"); class Dune { constructor(credentials = {}) { let { password, username } = credentials; password !== null && password !== void 0 ? password : (password = process.env.DUNE_PASSWORD); username !== null && username !== void 0 ? username : (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_1.Cookies(); } getCsrfToken() { return tslib_1.__awaiter(this, void 0, void 0, function* () { const response = yield fetch(constants_1.URLS.CSRF, { method: 'POST' }); this.csrf = (yield response.json()).csrf; this.cookies.set(response); }); } getAuthCookies() { return tslib_1.__awaiter(this, void 0, void 0, function* () { yield fetch(constants_1.URLS.AUTH, { body: new URLSearchParams({ action: 'login', csrf: this.csrf, next: constants_1.URLS.BASE, password: this.password, username: this.username, }), headers: Object.assign(Object.assign({}, constants_1.HEADERS), { 'Content-Type': 'application/x-www-form-urlencoded', cookie: `csrf=${this.csrf}` }), method: 'POST', redirect: 'manual', }).then((res) => { this.cookies.set(res); }); }); } getAuthToken() { return tslib_1.__awaiter(this, void 0, void 0, function* () { const response = yield fetch(constants_1.URLS.SESSION, { body: 'false', headers: Object.assign(Object.assign({}, constants_1.HEADERS), { cookie: this.cookies.toString() }), method: 'POST', }); this.token = (yield response.json()).token; }); } login() { return tslib_1.__awaiter(this, void 0, void 0, function* () { yield this.getCsrfToken(); yield this.getAuthCookies(); yield this.getAuthToken(); this.loggedAt = new Date(); }); } executeQuery(queryId, parameters = []) { return tslib_1.__awaiter(this, void 0, void 0, function* () { const res = yield fetch(constants_1.URLS.GRAPH_EXEC_ID, { body: JSON.stringify(Object.assign(Object.assign({}, constants_1.EXECUTE_QUERY_BODY), { variables: { parameters, query_id: queryId } })), headers: Object.assign(Object.assign({}, constants_1.HEADERS), { authorization: `Bearer ${this.token}`, 'Content-Type': 'application/json' }), method: 'POST', }); this.executionId = (yield res.json()).data.execute_query_v2.job_id; }); } query(queryId, parameterDatas) { return tslib_1.__awaiter(this, void 0, void 0, function* () { const parameters = Parameters_1.Parameters.create(parameterDatas); yield this.executeQuery(queryId, parameters); let executionSucceeded = null; while (executionSucceeded === null) { const res = yield fetch(constants_1.URLS.GRAPH_QUERY, { body: JSON.stringify(Object.assign(Object.assign({}, constants_1.QUERY_BODY), { variables: { execution_id: this.executionId, parameters, query_id: queryId, } })), headers: Object.assign(Object.assign({}, constants_1.HEADERS), { authorization: `Bearer ${this.token}`, 'Content-Type': 'application/json' }), method: 'POST', }); executionSucceeded = (yield res.json()).data.get_execution .execution_succeeded; yield (0, utils_1.delay)(1500); } const { columns, data } = executionSucceeded; return { columns, data }; }); } } tslib_1.__decorate([ decorators_1.isCsrfPresent, tslib_1.__metadata("design:type", Function), tslib_1.__metadata("design:paramtypes", []), tslib_1.__metadata("design:returntype", Promise) ], Dune.prototype, "getAuthCookies", null); tslib_1.__decorate([ decorators_1.isCookiesPresent, tslib_1.__metadata("design:type", Function), tslib_1.__metadata("design:paramtypes", []), tslib_1.__metadata("design:returntype", Promise) ], Dune.prototype, "getAuthToken", null); tslib_1.__decorate([ decorators_1.isTokenPresent, tslib_1.__metadata("design:type", Function), tslib_1.__metadata("design:paramtypes", [Number, void 0]), tslib_1.__metadata("design:returntype", Promise) ], Dune.prototype, "executeQuery", null); tslib_1.__decorate([ (0, decorators_1.maybeLogin)(), tslib_1.__metadata("design:type", Function), tslib_1.__metadata("design:paramtypes", [Number, Array]), tslib_1.__metadata("design:returntype", Promise) ], Dune.prototype, "query", null); exports.Dune = Dune; exports.dune = new Dune(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Dune.js","sourceRoot":"src/","sources":["Dune.ts"],"names":[],"mappings":";;;;AAAA,2CAA2E;AAC3E,uCAAmC;AACnC,6CAKqB;AACrB,6CAAyD;AACzD,mCAA+B;AAE/B,MAAa,IAAI;IASf,YAAY,cAAwD,EAAE;QACpE,IAAI,EAAE,QAAQ,EAAE,QAAQ,EAAE,GAAG,WAAW,CAAA;QACxC,QAAQ,aAAR,QAAQ,cAAR,QAAQ,IAAR,QAAQ,GAAK,OAAO,CAAC,GAAG,CAAC,aAAa,EAAA;QACtC,QAAQ,aAAR,QAAQ,cAAR,QAAQ,IAAR,QAAQ,GAAK,OAAO,CAAC,GAAG,CAAC,aAAa,EAAA;QAEtC,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE;YACnD,IAAI,OAAO,KAAK,KAAK,QAAQ;gBAC3B,MAAM,IAAI,KAAK,CAAC,GAAG,GAAG,qBAAqB,CAAC,CAAA;YAC9C,IAAI,KAAK,KAAK,EAAE;gBAAE,MAAM,IAAI,KAAK,CAAC,GAAG,GAAG,+BAA+B,CAAC,CAAA;QAC1E,CAAC,CAAC,CAAA;QAGF,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAA;QAExB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAA;QACxB,IAAI,CAAC,OAAO,GAAG,IAAI,iBAAO,EAAE,CAAA;IAC9B,CAAC;IAEa,YAAY;;YACxB,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,gBAAI,CAAC,IAAI,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAA;YAE3D,IAAI,CAAC,IAAI,GAAG,CAAC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC,IAAI,CAAA;YACxC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAA;QAC5B,CAAC;KAAA;IAGa,cAAc;;YAC1B,MAAM,KAAK,CAAC,gBAAI,CAAC,IAAI,EAAE;gBAErB,IAAI,EAAE,IAAI,eAAe,CAAC;oBACxB,MAAM,EAAE,OAAO;oBACf,IAAI,EAAE,IAAI,CAAC,IAAI;oBACf,IAAI,EAAE,gBAAI,CAAC,IAAI;oBACf,QAAQ,EAAE,IAAI,CAAC,QAAQ;oBACvB,QAAQ,EAAE,IAAI,CAAC,QAAQ;iBACxB,CAAC;gBACF,OAAO,kCACF,mBAAO,KACV,cAAc,EAAE,mCAAmC,EAEnD,MAAM,EAAE,QAAQ,IAAI,CAAC,IAAI,EAAE,GAC5B;gBACD,MAAM,EAAE,MAAM;gBACd,QAAQ,EAAE,QAAQ;aACnB,CAAC,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE;gBACd,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;YACvB,CAAC,CAAC,CAAA;QACJ,CAAC;KAAA;IAGa,YAAY;;YACxB,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,gBAAI,CAAC,OAAO,EAAE;gBACzC,IAAI,EAAE,OAAO;gBACb,OAAO,kCACF,mBAAO,KACV,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,GAChC;gBACD,MAAM,EAAE,MAAM;aACf,CAAC,CAAA;YAEF,IAAI,CAAC,KAAK,GAAG,CAAC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC,KAAK,CAAA;QAC5C,CAAC;KAAA;IAEY,KAAK;;YAChB,MAAM,IAAI,CAAC,YAAY,EAAE,CAAA;YACzB,MAAM,IAAI,CAAC,cAAc,EAAE,CAAA;YAC3B,MAAM,IAAI,CAAC,YAAY,EAAE,CAAA;YACzB,IAAI,CAAC,QAAQ,GAAG,IAAI,IAAI,EAAE,CAAA;QAC5B,CAAC;KAAA;IAGa,YAAY,CACxB,OAAe,EACf,aAAmD,EAAE;;YAErD,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,gBAAI,CAAC,aAAa,EAAE;gBAC1C,IAAI,EAAE,IAAI,CAAC,SAAS,iCACf,8BAAkB,KACrB,SAAS,EAAE,EAAE,UAAU,EAAE,QAAQ,EAAE,OAAO,EAAE,IAC5C;gBACF,OAAO,kCACF,mBAAO,KAEV,aAAa,EAAE,UAAU,IAAI,CAAC,KAAK,EAAE,EACrC,cAAc,EAAE,kBAAkB,GACnC;gBACD,MAAM,EAAE,MAAM;aACf,CAAC,CAAA;YAEF,IAAI,CAAC,WAAW,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAA;QACpE,CAAC;KAAA;IAGY,KAAK,CAAC,OAAe,EAAE,cAA+B;;YACjE,MAAM,UAAU,GAAG,uBAAU,CAAC,MAAM,CAAC,cAAc,CAAC,CAAA;YACpD,MAAM,IAAI,CAAC,YAAY,CAAC,OAAO,EAAE,UAAU,CAAC,CAAA;YAE5C,IAAI,kBAAkB,GAA8C,IAAI,CAAA;YAExE,OAAO,kBAAkB,KAAK,IAAI,EAAE;gBAClC,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,gBAAI,CAAC,WAAW,EAAE;oBACxC,IAAI,EAAE,IAAI,CAAC,SAAS,iCACf,sBAAU,KACb,SAAS,EAAE;4BACT,YAAY,EAAE,IAAI,CAAC,WAAW;4BAC9B,UAAU;4BACV,QAAQ,EAAE,OAAO;yBAClB,IACD;oBACF,OAAO,kCACF,mBAAO,KAEV,aAAa,EAAE,UAAU,IAAI,CAAC,KAAK,EAAE,EACrC,cAAc,EAAE,kBAAkB,GACnC;oBACD,MAAM,EAAE,MAAM;iBACf,CAAC,CAAA;gBACF,kBAAkB,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,aAAa;qBACvD,mBAAmB,CAAA;gBAEtB,MAAM,IAAA,aAAK,EAAC,IAAI,CAAC,CAAA;aAClB;YAED,MAAM,EAAE,OAAO,EAAE,IAAI,EAAE,GAAG,kBAAkB,CAAA;YAC5C,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAA;QAC1B,CAAC;KAAA;CACF;AApGe;IADb,0BAAa;;;;0CAsBb;AAGa;IADb,6BAAgB;;;;wCAYhB;AAUa;IADb,2BAAc;;;;wCAoBd;AAGY;IADZ,IAAA,uBAAU,GAAE;;;;iCAiCZ;AAtIH,oBAuIC;AAEY,QAAA,IAAI,GAAG,IAAI,IAAI,EAAE,CAAA","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"]}