dune-ts
Version:
Unofficial Dune Analytics Typescript Client
128 lines • 16.2 kB
JavaScript
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":"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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"]}
;