nocobase/packages/core/database/src/query-interface/sqlite-query-interface.ts
2025-04-02 15:52:55 +08:00

158 lines
4.1 KiB
TypeScript

/**
* This file is part of the NocoBase (R) project.
* Copyright (c) 2020-2024 NocoBase Co., Ltd.
* Authors: NocoBase Team.
*
* This project is dual-licensed under AGPL-3.0 and NocoBase Commercial License.
* For more information, please refer to: https://www.nocobase.com/agreement.
*/
/* istanbul ignore file -- @preserve */
import { Collection } from '../collection';
import sqlParser from '../sql-parser';
import QueryInterface, { ChangeColumnOptions, TableInfo } from './query-interface';
import { Transaction } from 'sequelize';
export default class SqliteQueryInterface extends QueryInterface {
constructor(db) {
super(db, { changeColumnMode: 'sequelize' });
}
async collectionTableExists(collection: Collection, options?) {
const transaction = options?.transaction;
const tableName = collection.model.tableName;
const sql = `SELECT name
FROM sqlite_master
WHERE type = 'table'
AND name = '${tableName}';`;
const results = await this.db.sequelize.query(sql, { type: 'SELECT', transaction });
return results.length > 0;
}
async listViews() {
const sql = `
SELECT name, sql as definition
FROM sqlite_master
WHERE type = 'view'
ORDER BY name;
`;
return await this.db.sequelize.query(sql, {
type: 'SELECT',
});
}
async viewColumnUsage(options: { viewName: string; schema?: string }): Promise<{
[view_column_name: string]: {
column_name: string;
table_name: string;
table_schema?: string;
};
}> {
try {
const { ast } = this.parseSQL(await this.viewDef(options.viewName));
const columns = ast.columns;
const results = [];
for (const column of columns) {
if (column.expr.type === 'column_ref') {
results.push([
column.as || column.expr.column,
{
column_name: column.expr.column,
table_name: column.expr.table,
},
]);
}
}
return Object.fromEntries(results);
} catch (e) {
this.db.logger.warn(e);
return {};
}
}
parseSQL(sql: string): any {
return sqlParser.parse(sql);
}
async viewDef(viewName: string): Promise<string> {
const viewDefinition = await this.db.sequelize.query(
`SELECT sql
FROM sqlite_master
WHERE name = '${viewName}' AND type = 'view'`,
{
type: 'SELECT',
},
);
const createView = viewDefinition[0]['sql'];
const regex = /(?<=AS\s)([\s\S]*)/i;
const match = createView.match(regex);
const sql = match[0];
return sql;
}
showTableDefinition(tableInfo: TableInfo): Promise<any> {
return Promise.resolve(undefined);
}
async getAutoIncrementInfo(options: { tableInfo: TableInfo; fieldName: string; transaction: Transaction }): Promise<{
seqName?: string;
currentVal: number;
}> {
const { tableInfo, transaction } = options;
const tableName = tableInfo.tableName;
const sql = `SELECT seq
FROM sqlite_sequence
WHERE name = '${tableName}';`;
const results = await this.db.sequelize.query(sql, { type: 'SELECT', transaction });
const row = results[0];
if (!row) {
return {
currentVal: 0,
};
}
return {
currentVal: row['seq'],
};
}
async setAutoIncrementVal(options: {
tableInfo: TableInfo;
columnName: string;
seqName?: string;
currentVal: number;
transaction?: Transaction;
}): Promise<void> {
const { tableInfo, columnName, seqName, currentVal, transaction } = options;
const tableName = tableInfo.tableName;
const sql = `UPDATE sqlite_sequence
SET seq = ${currentVal}
WHERE name = '${tableName}';`;
await this.db.sequelize.query(sql, { transaction });
}
public generateJoinOnForJSONArray(left: string, right: string) {
return this.db.sequelize.literal(`${left} in (SELECT value from json_each(${right}))`);
}
changeColumnDefaultValueSQL(options: ChangeColumnOptions): Promise<string> {
return null;
}
}