teableio/teable

View on GitHub
apps/nestjs-backend/src/features/view/open-api/view-open-api.service.ts

Summary

Maintainability
D
2 days
Test Coverage
import {
  BadRequestException,
  Injectable,
  Logger,
  NotFoundException,
  ForbiddenException,
} from '@nestjs/common';
import type {
  IOtOperation,
  IViewRo,
  IViewVo,
  IColumnMetaRo,
  IViewPropertyKeys,
  IViewOptions,
  IGridColumnMeta,
  IFilter,
  IFilterItem,
  ILinkFieldOptions,
  IPluginViewOptions,
} from '@teable/core';
import {
  ViewType,
  IManualSortRo,
  ViewOpBuilder,
  generateShareId,
  VIEW_JSON_KEYS,
  validateOptionsType,
  FieldType,
  IdPrefix,
  generatePluginInstallId,
  generateOperationId,
} from '@teable/core';
import { PrismaService } from '@teable/db-main-prisma';
import { PluginPosition, PluginStatus } from '@teable/openapi';
import type {
  IViewPluginUpdateStorageRo,
  IGetViewFilterLinkRecordsVo,
  IUpdateOrderRo,
  IUpdateRecordOrdersRo,
  IViewInstallPluginRo,
  IViewShareMetaRo,
} from '@teable/openapi';
import { Knex } from 'knex';
import { InjectModel } from 'nest-knexjs';
import { ClsService } from 'nestjs-cls';
import { InjectDbProvider } from '../../../db-provider/db.provider';
import { IDbProvider } from '../../../db-provider/db.provider.interface';
import { EventEmitterService } from '../../../event-emitter/event-emitter.service';
import { Events } from '../../../event-emitter/events';
import type { IClsStore } from '../../../types/cls';
import { Timing } from '../../../utils/timing';
import { updateMultipleOrders, updateOrder } from '../../../utils/update-order';
import { FieldViewSyncService } from '../../field/field-calculate/field-view-sync.service';
import { FieldService } from '../../field/field.service';
import type { IFieldInstance } from '../../field/model/factory';
import { createFieldInstanceByRaw, createFieldInstanceByVo } from '../../field/model/factory';
import { RecordService } from '../../record/record.service';
import { createViewInstanceByRaw } from '../model/factory';
import { ViewService } from '../view.service';

@Injectable()
export class ViewOpenApiService {
  private logger = new Logger(ViewOpenApiService.name);

  constructor(
    private readonly prismaService: PrismaService,
    private readonly recordService: RecordService,
    private readonly viewService: ViewService,
    private readonly fieldService: FieldService,
    private readonly fieldViewSyncService: FieldViewSyncService,
    private readonly eventEmitterService: EventEmitterService,
    private readonly cls: ClsService<IClsStore>,
    @InjectDbProvider() private readonly dbProvider: IDbProvider,
    @InjectModel('CUSTOM_KNEX') private readonly knex: Knex
  ) {}

  async createView(tableId: string, viewRo: IViewRo) {
    if (viewRo.type === ViewType.Plugin) {
      const res = await this.pluginInstall(tableId, {
        name: viewRo.name,
        pluginId: (viewRo.options as IPluginViewOptions).pluginId,
      });
      return this.viewService.getViewById(res.viewId);
    }
    return await this.prismaService.$tx(async () => {
      return this.createViewInner(tableId, viewRo);
    });
  }

  async deleteView(tableId: string, viewId: string, windowId?: string) {
    const result = await this.prismaService.$tx(async () => {
      await this.fieldViewSyncService.deleteLinkOptionsDependenciesByViewId(tableId, viewId);
      return await this.deleteViewInner(tableId, viewId);
    });

    this.eventEmitterService.emitAsync(Events.OPERATION_VIEW_DELETE, {
      operationId: generateOperationId(),
      windowId,
      tableId,
      viewId,
      userId: this.cls.get('user.id'),
    });

    return result;
  }

  private async createViewInner(tableId: string, viewRo: IViewRo): Promise<IViewVo> {
    return await this.viewService.createView(tableId, viewRo);
  }

  private async deleteViewInner(tableId: string, viewId: string) {
    return await this.viewService.deleteView(tableId, viewId);
  }

  private updateRecordOrderSql(orderRawSql: string, dbTableName: string, indexField: string) {
    return this.knex
      .raw(
        `
        UPDATE :dbTableName:
        SET :indexField: = temp_order.new_order
        FROM (
          SELECT __id, ROW_NUMBER() OVER (ORDER BY ${orderRawSql}) AS new_order FROM :dbTableName:
        ) AS temp_order
        WHERE :dbTableName:.__id = temp_order.__id AND :dbTableName:.:indexField: != temp_order.new_order;
      `,
        {
          dbTableName,
          indexField,
        }
      )
      .toQuery();
  }

  @Timing()
  async manualSort(tableId: string, viewId: string, viewOrderRo: IManualSortRo) {
    const { sortObjs } = viewOrderRo;
    const dbTableName = await this.recordService.getDbTableName(tableId);
    const fields = await this.fieldService.getFieldsByQuery(tableId, { viewId });
    const indexField = await this.viewService.getOrCreateViewIndexField(dbTableName, viewId);

    const queryBuilder = this.knex(dbTableName);

    const fieldInsMap = fields.reduce(
      (map, field) => {
        map[field.id] = createFieldInstanceByVo(field);
        return map;
      },
      {} as Record<string, IFieldInstance>
    );

    const orderRawSql = this.dbProvider
      .sortQuery(queryBuilder, fieldInsMap, sortObjs)
      .getRawSortSQLText();

    // build ops
    const newSort = {
      sortObjs: sortObjs,
      manualSort: true,
    };

    await this.prismaService.$tx(async (prisma) => {
      await prisma.$executeRawUnsafe(
        this.updateRecordOrderSql(orderRawSql, dbTableName, indexField)
      );
      await this.viewService.updateViewSort(tableId, viewId, newSort);
    });
  }

  async updateViewColumnMeta(
    tableId: string,
    viewId: string,
    columnMetaRo: IColumnMetaRo,
    windowId?: string
  ) {
    const view = await this.prismaService.view
      .findFirstOrThrow({
        where: { tableId, id: viewId },
        select: {
          columnMeta: true,
          version: true,
          id: true,
          type: true,
        },
      })
      .catch(() => {
        throw new BadRequestException('view found column meta error');
      });

    // validate field legal
    const fields = await this.prismaService.field.findMany({
      where: { tableId, deletedTime: null },
      select: {
        id: true,
        isPrimary: true,
      },
    });
    const primaryFields = fields.filter((field) => field.isPrimary).map((field) => field.id);

    const isHiddenPrimaryField = columnMetaRo.some(
      (f) => primaryFields.includes(f.fieldId) && (f.columnMeta as IGridColumnMeta).hidden
    );
    const fieldIds = columnMetaRo.map(({ fieldId }) => fieldId);

    if (!fieldIds.every((id) => fields.map(({ id }) => id).includes(id))) {
      throw new BadRequestException('field is not found in table');
    }

    const allowHiddenPrimaryType = [ViewType.Calendar, ViewType.Form];
    /**
     * validate whether hidden primary field
     * only form view or list view(todo) can hidden primary field
     */
    if (isHiddenPrimaryField && !allowHiddenPrimaryType.includes(view.type as ViewType)) {
      throw new ForbiddenException('primary field can not be hidden');
    }

    const curColumnMeta = JSON.parse(view.columnMeta);
    const ops: IOtOperation[] = [];

    columnMetaRo.forEach(({ fieldId, columnMeta }) => {
      const obj = {
        fieldId,
        newColumnMeta: { ...curColumnMeta[fieldId], ...columnMeta },
        oldColumnMeta: curColumnMeta[fieldId] ? curColumnMeta[fieldId] : undefined,
      };
      ops.push(ViewOpBuilder.editor.updateViewColumnMeta.build(obj));
    });

    await this.updateViewByOps(tableId, viewId, ops);

    if (windowId) {
      this.eventEmitterService.emitAsync(Events.OPERATION_VIEW_UPDATE, {
        tableId,
        windowId,
        viewId,
        userId: this.cls.get('user.id'),
        byOps: ops,
      });
    }
  }

  async updateShareMeta(tableId: string, viewId: string, viewShareMetaRo: IViewShareMetaRo) {
    return this.setViewProperty(tableId, viewId, 'shareMeta', viewShareMetaRo);
  }

  async setViewProperty(
    tableId: string,
    viewId: string,
    key: IViewPropertyKeys,
    newValue: unknown,
    windowId?: string
  ) {
    const curView = await this.prismaService.view
      .findFirstOrThrow({
        select: { [key]: true },
        where: { tableId, id: viewId, deletedTime: null },
      })
      .catch(() => {
        throw new BadRequestException('View not found');
      });
    const oldValue =
      curView[key] != null && VIEW_JSON_KEYS.includes(key)
        ? JSON.parse(curView[key])
        : curView[key];
    const ops = ViewOpBuilder.editor.setViewProperty.build({
      key,
      newValue,
      oldValue,
    });

    await this.updateViewByOps(tableId, viewId, [ops]);

    if (windowId) {
      this.eventEmitterService.emitAsync(Events.OPERATION_VIEW_UPDATE, {
        tableId,
        windowId,
        viewId,
        userId: this.cls.get('user.id'),
        byKey: {
          key,
          newValue,
          oldValue,
        },
      });
    }
  }

  async updateViewByOps(tableId: string, viewId: string, ops: IOtOperation[]) {
    return await this.prismaService.$tx(async () => {
      return await this.viewService.updateViewByOps(tableId, viewId, ops);
    });
  }

  async patchViewOptions(
    tableId: string,
    viewId: string,
    viewOptions: IViewOptions,
    windowId?: string
  ) {
    const curView = await this.prismaService.view
      .findFirstOrThrow({
        select: { options: true, type: true },
        where: { tableId, id: viewId, deletedTime: null },
      })
      .catch(() => {
        throw new BadRequestException('View option not found');
      });
    const { options, type: viewType } = curView;

    // validate option type
    try {
      validateOptionsType(viewType as ViewType, viewOptions);
    } catch (err) {
      throw new BadRequestException(err);
    }

    const oldOptions = options ? JSON.parse(options) : options;
    const op = ViewOpBuilder.editor.setViewProperty.build({
      key: 'options',
      newValue: {
        ...oldOptions,
        ...viewOptions,
      },
      oldValue: oldOptions,
    });
    await this.updateViewByOps(tableId, viewId, [op]);

    if (windowId) {
      this.eventEmitterService.emitAsync(Events.OPERATION_VIEW_UPDATE, {
        tableId,
        windowId,
        viewId,
        userId: this.cls.get('user.id'),
        byOps: [op],
      });
    }
  }

  /**
   * shuffle view order
   */
  async shuffle(tableId: string) {
    const views = await this.prismaService.view.findMany({
      where: { tableId, deletedTime: null },
      select: { id: true, order: true },
      orderBy: { order: 'asc' },
    });

    this.logger.log(`lucky view shuffle! ${tableId}`, 'shuffle');

    await this.prismaService.$tx(async () => {
      for (let i = 0; i < views.length; i++) {
        const view = views[i];
        await this.viewService.updateViewByOps(tableId, view.id, [
          ViewOpBuilder.editor.setViewProperty.build({
            key: 'order',
            newValue: i,
            oldValue: view.order,
          }),
        ]);
      }
    });
  }

  async updateViewOrder(
    tableId: string,
    viewId: string,
    orderRo: IUpdateOrderRo,
    windowId?: string
  ) {
    const { anchorId, position } = orderRo;

    const view = await this.prismaService.view
      .findFirstOrThrow({
        select: { order: true, id: true },
        where: { tableId, id: viewId, deletedTime: null },
      })
      .catch(() => {
        throw new NotFoundException(`View ${viewId} not found in the table`);
      });

    const anchorView = await this.prismaService.view
      .findFirstOrThrow({
        select: { order: true, id: true },
        where: { tableId, id: anchorId, deletedTime: null },
      })
      .catch(() => {
        throw new NotFoundException(`Anchor ${anchorId} not found in the table`);
      });

    await updateOrder({
      query: tableId,
      position,
      item: view,
      anchorItem: anchorView,
      getNextItem: async (whereOrder, align) => {
        return this.prismaService.view.findFirst({
          select: { order: true, id: true },
          where: {
            tableId,
            deletedTime: null,
            order: whereOrder,
          },
          orderBy: { order: align },
        });
      },
      update: async (
        parentId: string,
        id: string,
        data: { newOrder: number; oldOrder: number }
      ) => {
        const op = ViewOpBuilder.editor.setViewProperty.build({
          key: 'order',
          newValue: data.newOrder,
          oldValue: data.oldOrder,
        });
        await this.updateViewByOps(parentId, id, [op]);

        if (windowId) {
          this.eventEmitterService.emitAsync(Events.OPERATION_VIEW_UPDATE, {
            tableId,
            windowId,
            viewId,
            userId: this.cls.get('user.id'),
            byOps: [op],
          });
        }
      },
      shuffle: this.shuffle.bind(this),
    });
  }

  /**
   * shuffle record order
   */
  async shuffleRecords(dbTableName: string, indexField: string) {
    const recordCount = await this.recordService.getAllRecordCount(dbTableName);
    if (recordCount > 100_000) {
      throw new BadRequestException('Not enough gap to move the row here');
    }

    const sql = this.updateRecordOrderSql(
      this.knex.raw(`?? ASC`, [indexField]).toQuery(),
      dbTableName,
      indexField
    );

    await this.prismaService.$executeRawUnsafe(sql);
  }

  async updateRecordOrdersInner(props: {
    tableId: string;
    dbTableName: string;
    itemLength: number;
    indexField: string;
    orderRo: {
      anchorId: string;
      position: 'before' | 'after';
    };
    update: (indexes: number[]) => Promise<void>;
  }) {
    const { tableId, itemLength, dbTableName, indexField, orderRo, update } = props;
    const { anchorId, position } = orderRo;

    const anchorRecordSql = this.knex(dbTableName)
      .select({
        id: '__id',
        order: indexField,
      })
      .where('__id', anchorId)
      .toQuery();

    const anchorRecord = await this.prismaService
      .txClient()
      .$queryRawUnsafe<{ id: string; order: number }[]>(anchorRecordSql)
      .then((res) => {
        return res[0];
      });

    if (!anchorRecord) {
      throw new NotFoundException(`Anchor ${anchorId} not found in the table`);
    }

    await updateMultipleOrders({
      parentId: tableId,
      position,
      itemLength,
      anchorItem: anchorRecord,
      getNextItem: async (whereOrder, align) => {
        const nextRecordSql = this.knex(dbTableName)
          .select({
            id: '__id',
            order: indexField,
          })
          .where(
            indexField,
            whereOrder.lt != null ? '<' : '>',
            (whereOrder.lt != null ? whereOrder.lt : whereOrder.gt) as number
          )
          .orderBy(indexField, align)
          .limit(1)
          .toQuery();
        return this.prismaService
          .txClient()
          .$queryRawUnsafe<{ id: string; order: number }[]>(nextRecordSql)
          .then((res) => {
            return res[0];
          });
      },
      update,
      shuffle: async () => {
        await this.shuffleRecords(dbTableName, indexField);
      },
    });
  }

  async updateRecordIndexes(
    tableId: string,
    viewId: string,
    recordsWithOrder: {
      id: string;
      order?: Record<string, number>;
    }[]
  ) {
    // for notify view update only
    await this.prismaService.$tx(async () => {
      const ops = ViewOpBuilder.editor.setViewProperty.build({
        key: 'lastModifiedTime',
        newValue: new Date().toISOString(),
      });
      await this.viewService.updateViewByOps(tableId, viewId, [ops]);
      await this.recordService.updateRecordIndexes(tableId, recordsWithOrder);
    });
  }

  async updateRecordOrders(
    tableId: string,
    viewId: string,
    orderRo: IUpdateRecordOrdersRo,
    windowId?: string
  ) {
    const recordIds = orderRo.recordIds;
    const dbTableName = await this.recordService.getDbTableName(tableId);
    const orderIndexesBefore = windowId
      ? await this.recordService.getRecordIndexes(tableId, recordIds, viewId)
      : undefined;

    const indexField = await this.viewService.getOrCreateViewIndexField(dbTableName, viewId);

    await this.updateRecordOrdersInner({
      tableId,
      dbTableName,
      itemLength: recordIds.length,
      indexField,
      orderRo,
      update: async (indexes) => {
        // for notify view update only
        const ops = ViewOpBuilder.editor.setViewProperty.build({
          key: 'lastModifiedTime',
          newValue: new Date().toISOString(),
        });

        await this.prismaService.$tx(async (prisma) => {
          await this.viewService.updateViewByOps(tableId, viewId, [ops]);
          for (let i = 0; i < recordIds.length; i++) {
            const recordId = recordIds[i];
            const updateRecordSql = this.knex(dbTableName)
              .update({
                [indexField]: indexes[i],
              })
              .where('__id', recordId)
              .toQuery();
            await prisma.$executeRawUnsafe(updateRecordSql);
          }
        });
      },
    });

    if (windowId) {
      const orderIndexesAfter = await this.recordService.getRecordIndexes(
        tableId,
        recordIds,
        viewId
      );
      this.eventEmitterService.emitAsync(Events.OPERATION_RECORDS_ORDER_UPDATE, {
        tableId,
        windowId,
        recordIds,
        viewId,
        userId: this.cls.get('user.id'),
        orderIndexesBefore,
        orderIndexesAfter,
      });
    }
  }

  async refreshShareId(tableId: string, viewId: string) {
    const view = await this.prismaService.view.findUnique({
      where: { id: viewId, tableId, deletedTime: null },
      select: { shareId: true, enableShare: true },
    });
    if (!view) {
      throw new NotFoundException(`View ${viewId} does not exist`);
    }
    const { enableShare } = view;
    if (!enableShare) {
      throw new BadRequestException(`View ${viewId} has not been enabled share`);
    }
    const newShareId = generateShareId();
    const setShareIdOp = ViewOpBuilder.editor.setViewProperty.build({
      key: 'shareId',
      newValue: newShareId,
      oldValue: view.shareId || undefined,
    });
    await this.updateViewByOps(tableId, viewId, [setShareIdOp]);
    return { shareId: newShareId };
  }

  async enableShare(tableId: string, viewId: string) {
    const view = await this.prismaService.view.findUnique({
      where: { id: viewId, tableId, deletedTime: null },
    });
    if (!view) {
      throw new NotFoundException(`View ${viewId} does not exist`);
    }
    const { enableShare, shareId } = view;
    if (enableShare) {
      throw new BadRequestException(`View ${viewId} has already been enabled share`);
    }
    const newShareId = generateShareId();
    const enableShareOp = ViewOpBuilder.editor.setViewProperty.build({
      key: 'enableShare',
      newValue: true,
      oldValue: enableShare || undefined,
    });
    const setShareIdOp = ViewOpBuilder.editor.setViewProperty.build({
      key: 'shareId',
      newValue: newShareId,
      oldValue: shareId || undefined,
    });

    const ops = [enableShareOp, setShareIdOp];

    const viewInstance = createViewInstanceByRaw(view);
    if (!view.shareMeta && viewInstance.defaultShareMeta) {
      const initShareMetaOp = ViewOpBuilder.editor.setViewProperty.build({
        key: 'shareMeta',
        newValue: viewInstance.defaultShareMeta,
      });
      ops.push(initShareMetaOp);
    }
    await this.updateViewByOps(tableId, viewId, ops);
    return { shareId: newShareId };
  }

  async disableShare(tableId: string, viewId: string) {
    const view = await this.prismaService.view.findUnique({
      where: { id: viewId, tableId, deletedTime: null },
      select: { shareId: true, enableShare: true, shareMeta: true },
    });
    if (!view) {
      throw new NotFoundException(`View ${viewId} does not exist`);
    }
    const { enableShare } = view;
    if (!enableShare) {
      throw new BadRequestException(`View ${viewId} has already been disable share`);
    }
    const enableShareOp = ViewOpBuilder.editor.setViewProperty.build({
      key: 'enableShare',
      newValue: false,
      oldValue: enableShare || undefined,
    });

    await this.updateViewByOps(tableId, viewId, [enableShareOp]);
  }

  /**
   * @param linkFields {fieldId: foreignTableId}
   * @returns {foreignTableId: Set<recordId>}
   */
  private collectFilterLinkFieldRecords(linkFields: Record<string, string>, filter?: IFilter) {
    if (!filter || !filter.filterSet) {
      return undefined;
    }

    const tableRecordMap: Record<string, Set<string>> = {};

    const mergeRecordMap = (source: Record<string, Set<string>> = {}) => {
      for (const [fieldId, recordSet] of Object.entries(source)) {
        tableRecordMap[fieldId] = tableRecordMap[fieldId] || new Set();
        recordSet.forEach((item) => tableRecordMap[fieldId].add(item));
      }
    };

    for (const filterItem of filter.filterSet) {
      if ('filterSet' in filterItem) {
        const groupTableRecordMap = this.collectFilterLinkFieldRecords(
          linkFields,
          filterItem as IFilter
        );
        if (groupTableRecordMap) {
          mergeRecordMap(groupTableRecordMap);
        }
        continue;
      }

      const { value, fieldId } = filterItem as IFilterItem;

      const foreignTableId = linkFields[fieldId];
      if (!foreignTableId) {
        continue;
      }

      if (Array.isArray(value)) {
        mergeRecordMap({ [foreignTableId]: new Set(value as string[]) });
      } else if (typeof value === 'string' && value.startsWith(IdPrefix.Record)) {
        mergeRecordMap({ [foreignTableId]: new Set([value]) });
      }
    }

    return tableRecordMap;
  }

  async getFilterLinkRecords(tableId: string, viewId: string) {
    const view = await this.viewService.getViewById(viewId);
    return this.getFilterLinkRecordsByTable(tableId, view.filter);
  }

  async getFilterLinkRecordsByTable(tableId: string, filter?: IFilter) {
    if (!filter) {
      return [];
    }
    const linkFields = await this.prismaService.field.findMany({
      where: { tableId, deletedTime: null, type: FieldType.Link },
    });
    const linkFieldTableMap = linkFields.reduce(
      (map, field) => {
        const { foreignTableId } = JSON.parse(field.options as string) as ILinkFieldOptions;
        map[field.id] = foreignTableId;
        return map;
      },
      {} as Record<string, string>
    );

    const tableRecordMap = this.collectFilterLinkFieldRecords(linkFieldTableMap, filter);

    if (!tableRecordMap) {
      return [];
    }
    const res: IGetViewFilterLinkRecordsVo = [];
    for (const [foreignTableId, recordSet] of Object.entries(tableRecordMap)) {
      const dbTableName = await this.recordService.getDbTableName(foreignTableId);
      const primaryField = await this.prismaService.field.findFirst({
        where: { tableId: foreignTableId, isPrimary: true, deletedTime: null },
      });
      if (!primaryField) {
        continue;
      }

      const dbFieldName = primaryField.dbFieldName;

      const nativeQuery = this.knex(dbTableName)
        .select('__id as id', `${dbFieldName} as title`)
        .orderBy('__auto_number')
        .whereIn('__id', Array.from(recordSet))
        .toQuery();

      const list = await this.prismaService
        .txClient()
        .$queryRawUnsafe<{ id: string; title: string | null }[]>(nativeQuery);
      const fieldInstances = createFieldInstanceByRaw(primaryField);
      res.push({
        tableId: foreignTableId,
        records: list.map(({ id, title }) => ({
          id,
          title:
            fieldInstances.cellValue2String(fieldInstances.convertDBValue2CellValue(title)) ||
            undefined,
        })),
      });
    }
    return res;
  }

  async pluginInstall(tableId: string, ro: IViewInstallPluginRo) {
    const userId = this.cls.get('user.id');
    const { name, pluginId } = ro;
    const plugin = await this.prismaService.plugin.findUnique({
      where: { id: pluginId, status: PluginStatus.Published },
      select: { id: true, name: true, logo: true, positions: true },
    });
    if (!plugin) {
      throw new NotFoundException(`Plugin ${pluginId} not found`);
    }
    if (!plugin.positions.includes(PluginPosition.View)) {
      throw new BadRequestException(`Plugin ${pluginId} does not support install in view`);
    }
    const viewName = name || plugin.name;
    return this.prismaService.$tx(async (prisma) => {
      const pluginInstallId = generatePluginInstallId();
      const view = await this.createViewInner(tableId, {
        name: viewName,
        type: ViewType.Plugin,
        options: {
          pluginInstallId,
          pluginId,
          pluginLogo: plugin.logo,
        } as IPluginViewOptions,
      });
      const table = await prisma.tableMeta.findUniqueOrThrow({
        where: { id: tableId, deletedTime: null },
        select: { baseId: true },
      });
      const newPlugin = await prisma.pluginInstall.create({
        data: {
          id: pluginInstallId,
          baseId: table?.baseId,
          positionId: view.id,
          position: PluginPosition.View,
          name: viewName,
          pluginId: ro.pluginId,
          createdBy: userId,
        },
      });
      return {
        pluginId: newPlugin.pluginId,
        pluginInstallId: newPlugin.id,
        name: newPlugin.name,
        viewId: view.id,
      };
    });
  }

  async updatePluginStorage(viewId: string, storage: IViewPluginUpdateStorageRo['storage']) {
    const pluginInstall = await this.prismaService.pluginInstall.findFirst({
      where: { positionId: viewId, position: PluginPosition.View },
      select: { id: true },
    });
    if (!pluginInstall) {
      throw new NotFoundException(`Plugin install not found`);
    }
    return this.prismaService.pluginInstall.update({
      where: { id: pluginInstall.id },
      data: { storage: JSON.stringify(storage) },
    });
  }

  async getPluginInstall(tableId: string, viewId: string) {
    const table = await this.prismaService.tableMeta.findUniqueOrThrow({
      where: { id: tableId, deletedTime: null },
      select: { baseId: true },
    });
    const pluginInstall = await this.prismaService.pluginInstall.findFirst({
      where: { positionId: viewId, position: PluginPosition.View },
      select: {
        id: true,
        pluginId: true,
        name: true,
        storage: true,
        plugin: {
          select: { url: true },
        },
      },
    });
    if (!pluginInstall) {
      throw new NotFoundException(`Plugin install not found`);
    }
    return {
      name: pluginInstall.name,
      pluginId: pluginInstall.pluginId,
      pluginInstallId: pluginInstall.id,
      storage: pluginInstall.storage ? JSON.parse(pluginInstall.storage) : undefined,
      baseId: table.baseId,
      url: pluginInstall.plugin.url || undefined,
    };
  }
}