get-graph-question.ts•3.82 kB
import { createAction, Property } from '@activepieces/pieces-framework';
import { metabaseAuth } from '../..';
import jwt from 'jsonwebtoken';
import { chromium } from 'playwright';
export const getGraphQuestion = createAction({
  name: 'getGraphQuestion',
  auth: metabaseAuth,
  requireAuth: true,
  displayName: 'Get the graph of the question',
  description: 'Get the graph of a Metabase question and save it as a png',
  props: {
    questionId: Property.ShortText({
      displayName: 'Metabase question ID',
      required: true,
    }),
    parameters: Property.Object({
      displayName: 'Parameters (slug name -> value)',
      required: false,
    }),
    graphName: Property.ShortText({
      displayName: 'The name of the graph (without the extension)',
      required: false,
    }),
    waitTime: Property.Number({
      displayName: 'Wait Time (seconds)',
      description:
        'How long to wait for the graph to render completely in seconds',
      required: true,
      defaultValue: 5,
    }),
  },
  async run({ auth, propsValue, files }) {
    if ('embeddingKey' in auth && !auth.embeddingKey)
      return 'An embedding key is needed.';
    if (propsValue.waitTime <= 0)
      return 'The wait time needs to be superior to 0';
    const questionId = propsValue.questionId.split('-')[0];
    const numericQuestionId = parseInt(questionId);
    const payload = {
      resource: { question: numericQuestionId },
      params: propsValue.parameters,
      exp: Math.round(Date.now() / 1000) + 10 * 60,
    };
    // @ts-expect-error we expect an embedding key if the user can use this action.
    const token = jwt.sign(payload, auth.embeddingKey);
    const graphName = propsValue.graphName
      ? propsValue.graphName + '.png'
      : `metabase_question_${questionId}.png`;
    const iframeUrl =
      auth.baseUrl + '/embed/question/' + token + '#bordered=true&titled=true';
    const browser = await chromium.launch({
      headless: true,
      chromiumSandbox: false,
      executablePath: '/usr/bin/chromium',
    });
    try {
      const context = await browser.newContext({
        viewport: {
          width: 1600,
          height: 1200,
        },
        deviceScaleFactor: 2,
      });
      const page = await context.newPage();
      const response = await page.goto(iframeUrl, {
        waitUntil: 'networkidle',
        timeout: 30000,
      });
      if (!response || !response.ok()) {
        throw new Error(
          `Page load failed with status: ${response ? response.status() : 400}`
        );
      }
      // we wait so the graph can load
      await page.waitForTimeout(propsValue.waitTime * 1000);
      const mainElement = await page.$('main');
      let screenshotBuffer;
      if (mainElement) {
        screenshotBuffer = await mainElement.screenshot({
          path: graphName,
          type: 'png',
        });
      } else {
        screenshotBuffer = await page.screenshot({
          path: graphName,
          type: 'png',
          clip: {
            x: 0,
            y: 0,
            width: 1600,
            height: 1120, // so it doesn't screenshot the metabase banner
          },
        });
      }
      const fileUrl = await files.write({
        fileName: graphName,
        data: screenshotBuffer,
      });
      return {
        file: {
          filename: graphName,
          base64Content: screenshotBuffer.toString('base64'),
          download: fileUrl,
        },
        iframeUrl,
      };
    } catch (error) {
      console.error(
        'Please verify that either your embedding key and question id are valid or that the question is embedded and published.'
      );
      console.error('Error capturing Metabase chart:', error);
      throw error;
    } finally {
      await browser.close();
    }
  },
});