mirror of
https://gitee.com/nocobase/nocobase.git
synced 2025-05-07 22:49:26 +08:00
162 lines
5.0 KiB
TypeScript
162 lines
5.0 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 { ModelStatic, Transaction, Transactionable } from 'sequelize';
|
|
import { Collection } from '../collection';
|
|
import sqlParser from '../sql-parser';
|
|
import QueryInterface, { ChangeColumnOptions, RemoveColumnOptions, TableInfo } from './query-interface';
|
|
|
|
export default class MysqlQueryInterface extends QueryInterface {
|
|
constructor(db) {
|
|
super(db, { changeColumnMode: 'sequelize' });
|
|
}
|
|
|
|
async collectionTableExists(collection: Collection, options?: Transactionable) {
|
|
const transaction = options?.transaction;
|
|
|
|
const tableName = collection.model.tableName;
|
|
const databaseName = this.db.options.database;
|
|
const sql = `SELECT TABLE_NAME
|
|
FROM INFORMATION_SCHEMA.TABLES
|
|
WHERE TABLE_SCHEMA = '${databaseName}'
|
|
AND TABLE_NAME = '${tableName}'`;
|
|
|
|
const results = await this.db.sequelize.query(sql, { type: 'SELECT', transaction });
|
|
return results.length > 0;
|
|
}
|
|
|
|
async listViews() {
|
|
const sql = `SELECT TABLE_NAME as name, VIEW_DEFINITION as definition
|
|
FROM information_schema.views
|
|
WHERE TABLE_SCHEMA = DATABASE()
|
|
ORDER BY TABLE_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(`SHOW CREATE VIEW ${viewName}`, { type: 'SELECT' });
|
|
const createView = viewDefinition[0]['Create View'];
|
|
const regex = /(?<=AS\s)([\s\S]*)/i;
|
|
const match = createView.match(regex);
|
|
const sql = match[0];
|
|
return sql;
|
|
}
|
|
|
|
async showTableDefinition(tableInfo: TableInfo): Promise<any> {
|
|
const { tableName } = tableInfo;
|
|
|
|
const sql = `SHOW CREATE TABLE ${this.db.utils.quoteTable(tableName)}`;
|
|
|
|
const results = await this.db.sequelize.query(sql, { type: 'SELECT' });
|
|
|
|
return results[0]['Create Table'];
|
|
}
|
|
|
|
async getAutoIncrementInfo(options: { tableInfo: TableInfo; fieldName: string; transaction: Transaction }): Promise<{
|
|
seqName?: string;
|
|
currentVal: number;
|
|
}> {
|
|
const { tableInfo, fieldName, transaction } = options;
|
|
|
|
const sql = `SELECT AUTO_INCREMENT as currentVal
|
|
FROM information_schema.tables
|
|
WHERE table_schema = DATABASE()
|
|
AND table_name = '${tableInfo.tableName}';`;
|
|
|
|
const results = await this.db.sequelize.query(sql, { type: 'SELECT', transaction });
|
|
|
|
let currentVal = results[0]['currentVal'] as number;
|
|
|
|
if (currentVal === null) {
|
|
// use max value of field instead
|
|
const maxSql = `SELECT MAX(\`${fieldName}\`) as currentVal
|
|
FROM \`${tableInfo.tableName}\`;`;
|
|
|
|
const maxResults = await this.db.sequelize.query(maxSql, { type: 'SELECT', transaction });
|
|
currentVal = maxResults[0]['currentVal'] as number;
|
|
}
|
|
|
|
return {
|
|
currentVal,
|
|
};
|
|
}
|
|
|
|
async setAutoIncrementVal(options: {
|
|
tableInfo: TableInfo;
|
|
columnName: string;
|
|
seqName?: string;
|
|
currentVal: number;
|
|
transaction?: Transaction;
|
|
}): Promise<void> {
|
|
const { tableInfo, columnName, seqName, currentVal, transaction } = options;
|
|
|
|
if (currentVal) {
|
|
const sql = `ALTER TABLE ${this.quoteIdentifier(tableInfo.tableName)} AUTO_INCREMENT = ${currentVal};`;
|
|
await this.db.sequelize.query(sql, { transaction });
|
|
}
|
|
}
|
|
|
|
public generateJoinOnForJSONArray(left: string, right: string) {
|
|
return this.db.sequelize.literal(`JSON_CONTAINS(${right}, JSON_ARRAY(${left}))`);
|
|
}
|
|
|
|
changeColumnDefaultValueSQL(options: ChangeColumnOptions): Promise<string> {
|
|
return null;
|
|
}
|
|
|
|
async beforeRemoveColumn(options: RemoveColumnOptions): Promise<void> {}
|
|
async afterRemoveColumn(options: RemoveColumnOptions): Promise<void> {}
|
|
|
|
nullSafe(): string {
|
|
return 'IFNULL';
|
|
}
|
|
|
|
async ensureSchema(schemaName: string): Promise<void> {}
|
|
}
|