Browse Source

Merge branch 'develop' into fix/signout

pull/5905/head
Wing-Kam Wong 1 year ago
parent
commit
0bde2eeb70
  1. 2
      .github/workflows/release-pr.yml
  2. 95
      packages/nc-gui/components/smartsheet/column/LinkedToAnotherRecordOptions.vue
  3. 5
      packages/nc-gui/composables/useColumnCreateStore.ts
  4. 6
      packages/nc-gui/composables/useTable.ts
  5. 240
      packages/nocodb/src/db/BaseModelSqlv2.ts
  6. 14
      packages/nocodb/src/db/sql-client/lib/sqlite/SqliteClient.ts
  7. 23
      packages/nocodb/src/db/sql-mgr/v2/ProjectMgrv2.ts
  8. 12
      packages/nocodb/src/db/sql-mgr/v2/SqlMgrv2.ts
  9. 1
      packages/nocodb/src/meta/meta.service.ts
  10. 1
      packages/nocodb/src/middlewares/extract-project-id/extract-project-id.middleware.ts
  11. 16
      packages/nocodb/src/models/Column.ts
  12. 6
      packages/nocodb/src/models/Model.ts
  13. 7
      packages/nocodb/src/models/Project.ts
  14. 10
      packages/nocodb/src/models/View.ts
  15. 2
      packages/nocodb/src/modules/global/init-meta-service.provider.ts
  16. 215
      packages/nocodb/src/services/columns.service.ts
  17. 12
      packages/nocodb/src/services/datas.service.ts
  18. 105
      packages/nocodb/src/services/tables.service.ts
  19. 2
      packages/nocodb/src/version-upgrader/NcUpgrader.ts
  20. 384
      packages/nocodb/src/version-upgrader/ncFilterUpgrader_0104004.ts
  21. 164
      packages/nocodb/src/version-upgrader/ncXcdbLTARUpgrader.ts
  22. 14
      packages/nocodb/tests/unit/rest/tests/tableRow.test.ts
  23. 24
      packages/nocodb/tests/unit/rest/tests/viewRow.test.ts
  24. 14
      tests/playwright/pages/Dashboard/Grid/index.ts
  25. 40
      tests/playwright/setup/xcdbProject.ts
  26. 324
      tests/playwright/tests/db/metaLTAR.spec.ts
  27. 25
      tests/playwright/tests/db/timezone.spec.ts

2
.github/workflows/release-pr.yml

@ -33,7 +33,7 @@ jobs:
# Get current PR number # Get current PR number
PR_NUMBER=${{github.event.number}} PR_NUMBER=${{github.event.number}}
# Get current version # Get current version
CURRENT_VERSION=$(curl -fs https://docs.nocodb.com/releases | grep article | grep div | grep h2 | grep 'id\="[^"]*' -o | cut -c 5- | cut -d\: -f1) CURRENT_VERSION=$(cat ./packages/nocodb/package.json | jq -r ".version")
# Construct tag name # Construct tag name
TAG_NAME=pr-${PR_NUMBER}-${CURRENT_DATE}-${CURRENT_TIME} TAG_NAME=pr-${PR_NUMBER}-${CURRENT_DATE}-${CURRENT_TIME}
echo "TARGET_TAG=${TAG_NAME}" >> $GITHUB_OUTPUT echo "TARGET_TAG=${TAG_NAME}" >> $GITHUB_OUTPUT

95
packages/nc-gui/components/smartsheet/column/LinkedToAnotherRecordOptions.vue

@ -16,7 +16,7 @@ const vModel = useVModel(props, 'value', emit)
const meta = $(inject(MetaInj, ref())) const meta = $(inject(MetaInj, ref()))
const { setAdditionalValidations, validateInfos, onDataTypeChange, sqlUi } = useColumnCreateStoreOrThrow() const { setAdditionalValidations, validateInfos, onDataTypeChange, sqlUi, isXcdbBase } = useColumnCreateStoreOrThrow()
const { tables } = $(storeToRefs(useProject())) const { tables } = $(storeToRefs(useProject()))
@ -86,54 +86,57 @@ const filterOption = (value: string, option: { key: string }) => option.key.toLo
</a-select> </a-select>
</a-form-item> </a-form-item>
</div> </div>
<template v-if="!isXcdbBase">
<div
class="text-xs cursor-pointer text-grey nc-more-options my-2 flex items-center gap-1 justify-end"
@click="advancedOptions = !advancedOptions"
>
{{ advancedOptions ? $t('general.hideAll') : $t('general.showMore') }}
<div <component :is="advancedOptions ? MdiMinusIcon : MdiPlusIcon" />
class="text-xs cursor-pointer text-grey nc-more-options my-2 flex items-center gap-1 justify-end"
@click="advancedOptions = !advancedOptions"
>
{{ advancedOptions ? $t('general.hideAll') : $t('general.showMore') }}
<component :is="advancedOptions ? MdiMinusIcon : MdiPlusIcon" />
</div>
<div v-if="advancedOptions" class="flex flex-col p-6 gap-4 border-2 mt-2">
<div class="flex flex-row space-x-2">
<a-form-item class="flex w-1/2" :label="$t('labels.onUpdate')">
<a-select
v-model:value="vModel.onUpdate"
:disabled="vModel.virtual"
name="onUpdate"
dropdown-class-name="nc-dropdown-on-update"
@change="onDataTypeChange"
>
<a-select-option v-for="(option, i) of onUpdateDeleteOptions" :key="i" :value="option">
{{ option }}
</a-select-option>
</a-select>
</a-form-item>
<a-form-item class="flex w-1/2" :label="$t('labels.onDelete')">
<a-select
v-model:value="vModel.onDelete"
:disabled="vModel.virtual"
name="onDelete"
dropdown-class-name="nc-dropdown-on-delete"
@change="onDataTypeChange"
>
<a-select-option v-for="(option, i) of onUpdateDeleteOptions" :key="i" :value="option">
{{ option }}
</a-select-option>
</a-select>
</a-form-item>
</div> </div>
<div class="flex flex-row"> <div v-if="advancedOptions" class="flex flex-col p-6 gap-4 border-2 mt-2">
<a-form-item> <div class="flex flex-row space-x-2">
<a-checkbox v-model:checked="vModel.virtual" :disabled="appInfo.isCloud" name="virtual" @change="onDataTypeChange" <a-form-item class="flex w-1/2" :label="$t('labels.onUpdate')">
>Virtual Relation</a-checkbox <a-select
> v-model:value="vModel.onUpdate"
</a-form-item> :disabled="vModel.virtual"
name="onUpdate"
dropdown-class-name="nc-dropdown-on-update"
@change="onDataTypeChange"
>
<a-select-option v-for="(option, i) of onUpdateDeleteOptions" :key="i" :value="option">
{{ option }}
</a-select-option>
</a-select>
</a-form-item>
<a-form-item class="flex w-1/2" :label="$t('labels.onDelete')">
<a-select
v-model:value="vModel.onDelete"
:disabled="vModel.virtual"
name="onDelete"
dropdown-class-name="nc-dropdown-on-delete"
@change="onDataTypeChange"
>
<a-select-option v-for="(option, i) of onUpdateDeleteOptions" :key="i" :value="option">
{{ option }}
</a-select-option>
</a-select>
</a-form-item>
</div>
<div class="flex flex-row">
<a-form-item>
<a-checkbox v-model:checked="vModel.virtual" :disabled="appInfo.isCloud" name="virtual"
@change="onDataTypeChange"
>Virtual Relation
</a-checkbox
>
</a-form-item>
</div>
</div> </div>
</div> </template>
</div> </div>
</template> </template>

5
packages/nc-gui/composables/useColumnCreateStore.ts

@ -31,7 +31,7 @@ interface ValidationsObj {
const [useProvideColumnCreateStore, useColumnCreateStore] = createInjectionState( const [useProvideColumnCreateStore, useColumnCreateStore] = createInjectionState(
(meta: Ref<TableType | undefined>, column: Ref<ColumnType | undefined>) => { (meta: Ref<TableType | undefined>, column: Ref<ColumnType | undefined>) => {
const projectStore = useProject() const projectStore = useProject()
const { isMysql: isMysqlFunc, isPg: isPgFunc, isMssql: isMssqlFunc } = projectStore const { isMysql: isMysqlFunc, isPg: isPgFunc, isMssql: isMssqlFunc, isXcdbBase: isXcdbBaseFunc } = projectStore
const { project, sqlUis } = storeToRefs(projectStore) const { project, sqlUis } = storeToRefs(projectStore)
const { $api } = useNuxtApp() const { $api } = useNuxtApp()
@ -52,6 +52,8 @@ const [useProvideColumnCreateStore, useColumnCreateStore] = createInjectionState
const isMssql = computed(() => isMssqlFunc(meta.value?.base_id ? meta.value?.base_id : Object.keys(sqlUis.value)[0])) const isMssql = computed(() => isMssqlFunc(meta.value?.base_id ? meta.value?.base_id : Object.keys(sqlUis.value)[0]))
const isXcdbBase = computed(() => isXcdbBaseFunc(meta.value?.base_id ? meta.value?.base_id : Object.keys(sqlUis.value)[0]))
const idType = null const idType = null
const additionalValidations = ref<ValidationsObj>({}) const additionalValidations = ref<ValidationsObj>({})
@ -289,6 +291,7 @@ const [useProvideColumnCreateStore, useColumnCreateStore] = createInjectionState
isMssql, isMssql,
isPg, isPg,
isMysql, isMysql,
isXcdbBase
} }
}, },
) )

6
packages/nc-gui/composables/useTable.ts

@ -30,7 +30,7 @@ export function useTable(onTableCreate?: (tableMeta: TableType) => void, baseId?
const { getMeta, removeMeta } = useMetas() const { getMeta, removeMeta } = useMetas()
const { loadTables } = useProject() const { loadTables, isXcdbBase } = useProject()
const { closeTab } = useTabs() const { closeTab } = useTabs()
const projectStore = useProject() const projectStore = useProject()
@ -88,7 +88,9 @@ export function useTable(onTableCreate?: (tableMeta: TableType) => void, baseId?
const meta = (await getMeta(table.id as string, true)) as TableType const meta = (await getMeta(table.id as string, true)) as TableType
const relationColumns = meta?.columns?.filter((c) => c.uidt === UITypes.LinkToAnotherRecord && !isSystemColumn(c)) const relationColumns = meta?.columns?.filter((c) => c.uidt === UITypes.LinkToAnotherRecord && !isSystemColumn(c))
if (relationColumns?.length) { // Check if table has any relation columns and show notification
// skip for xcdb base
if (relationColumns?.length && !isXcdbBase(table.base_id)) {
const refColMsgs = await Promise.all( const refColMsgs = await Promise.all(
relationColumns.map(async (c, i) => { relationColumns.map(async (c, i) => {
const refMeta = (await getMeta( const refMeta = (await getMeta(

240
packages/nocodb/src/db/BaseModelSqlv2.ts

@ -17,9 +17,19 @@ import Validator from 'validator';
import { customAlphabet } from 'nanoid'; import { customAlphabet } from 'nanoid';
import DOMPurify from 'isomorphic-dompurify'; import DOMPurify from 'isomorphic-dompurify';
import { v4 as uuidv4 } from 'uuid'; import { v4 as uuidv4 } from 'uuid';
import { Knex } from 'knex';
import { NcError } from '../helpers/catchError'; import { NcError } from '../helpers/catchError';
import getAst from '../helpers/getAst'; import getAst from '../helpers/getAst';
import { Audit, Column, Filter, Model, Project, Sort, View } from '../models'; import {
Audit,
Base,
Column,
Filter,
Model,
Project,
Sort,
View,
} from '../models';
import { sanitize, unsanitize } from '../helpers/sqlSanitize'; import { sanitize, unsanitize } from '../helpers/sqlSanitize';
import { import {
COMPARISON_OPS, COMPARISON_OPS,
@ -47,8 +57,8 @@ import type {
RollupColumn, RollupColumn,
SelectOption, SelectOption,
} from '../models'; } from '../models';
import type { Knex } from 'knex';
import type { SortType } from 'nocodb-sdk'; import type { SortType } from 'nocodb-sdk';
import Transaction = Knex.Transaction;
dayjs.extend(utc); dayjs.extend(utc);
dayjs.extend(timezone); dayjs.extend(timezone);
@ -1875,18 +1885,80 @@ class BaseModelSqlv2 {
} }
} }
async delByPk(id, trx?, cookie?) { async delByPk(id, _trx?, cookie?) {
let trx: Transaction = _trx;
try { try {
// retrieve data for handling params in hook // retrieve data for handling params in hook
const data = await this.readByPk(id); const data = await this.readByPk(id);
await this.beforeDelete(id, trx, cookie); await this.beforeDelete(id, trx, cookie);
const response = await this.dbDriver(this.tnPath)
.del() const execQueries: ((trx: Transaction) => Promise<any>)[] = [];
.where(await this._wherePk(id));
for (const column of this.model.columns) {
if (column.uidt !== UITypes.LinkToAnotherRecord) continue;
const colOptions =
await column.getColOptions<LinkToAnotherRecordColumn>();
switch (colOptions.type) {
case 'mm':
{
const mmTable = await Model.get(colOptions.fk_mm_model_id);
const mmParentColumn = await Column.get({
colId: colOptions.fk_mm_child_column_id,
});
execQueries.push((trx) =>
trx(mmTable.table_name)
.del()
.where(mmParentColumn.column_name, id),
);
}
break;
case 'hm':
{
// skip if it's an mm table column
const relatedTable = await colOptions.getRelatedTable();
if (relatedTable.mm) {
break;
}
const childColumn = await Column.get({
colId: colOptions.fk_child_column_id,
});
execQueries.push((trx) =>
trx(relatedTable.table_name)
.update({
[childColumn.column_name]: null,
})
.where(childColumn.column_name, id),
);
}
break;
case 'bt':
{
// nothing to do
}
break;
}
}
const where = await this._wherePk(id);
if (!trx) {
trx = await this.dbDriver.transaction();
}
await Promise.all(execQueries.map((q) => q(trx)));
const response = await trx(this.tnPath).del().where(where);
if (!_trx) await trx.commit();
await this.afterDelete(data, trx, cookie); await this.afterDelete(data, trx, cookie);
return response; return response;
} catch (e) { } catch (e) {
console.log(e); console.log(e);
if (!_trx) await trx.rollback();
await this.errorDelete(e, id, trx, cookie); await this.errorDelete(e, id, trx, cookie);
throw e; throw e;
} }
@ -2379,8 +2451,71 @@ class BaseModelSqlv2 {
res.push(d); res.push(d);
} }
const execQueries: ((trx: Transaction, ids: any[]) => Promise<any>)[] =
[];
const base = await Base.get(this.model.base_id);
for (const column of this.model.columns) {
if (column.uidt !== UITypes.LinkToAnotherRecord) continue;
const colOptions =
await column.getColOptions<LinkToAnotherRecordColumn>();
switch (colOptions.type) {
case 'mm':
{
const mmTable = await Model.get(colOptions.fk_mm_model_id);
const mmParentColumn = await Column.get({
colId: colOptions.fk_mm_child_column_id,
});
execQueries.push((trx, ids) =>
trx(mmTable.table_name)
.del()
.whereIn(mmParentColumn.column_name, ids),
);
}
break;
case 'hm':
{
// skip if it's an mm table column
const relatedTable = await colOptions.getRelatedTable();
if (relatedTable.mm) {
break;
}
const childColumn = await Column.get({
colId: colOptions.fk_child_column_id,
});
execQueries.push((trx, ids) =>
trx(relatedTable.table_name)
.update({
[childColumn.column_name]: null,
})
.whereIn(childColumn.column_name, ids),
);
}
break;
case 'bt':
{
// nothing to do
}
break;
}
}
const idsVals = res.map((d) => d[this.model.primaryKey.column_name]);
transaction = await this.dbDriver.transaction(); transaction = await this.dbDriver.transaction();
if (base.is_meta && execQueries.length > 0) {
for (const execQuery of execQueries) {
await execQuery(transaction, idsVals);
}
}
for (const d of res) { for (const d of res) {
await transaction(this.tnPath).del().where(d); await transaction(this.tnPath).del().where(d);
} }
@ -2401,6 +2536,7 @@ class BaseModelSqlv2 {
args: { where?: string; filterArr?: Filter[] } = {}, args: { where?: string; filterArr?: Filter[] } = {},
{ cookie }: { cookie?: any } = {}, { cookie }: { cookie?: any } = {},
) { ) {
let trx: Transaction;
try { try {
await this.model.getColumns(); await this.model.getColumns();
const { where } = this._getListArgs(args); const { where } = this._getListArgs(args);
@ -2425,14 +2561,102 @@ class BaseModelSqlv2 {
this.dbDriver, this.dbDriver,
); );
qb.del(); const execQueries: ((trx: Transaction, qb: any) => Promise<any>)[] = [];
for (const column of this.model.columns) {
if (column.uidt !== UITypes.LinkToAnotherRecord) continue;
const colOptions =
await column.getColOptions<LinkToAnotherRecordColumn>();
if (colOptions.type === 'bt') {
continue;
}
const count = (await qb) as any; const childColumn = await colOptions.getChildColumn();
const parentColumn = await colOptions.getParentColumn();
const parentTable = await parentColumn.getModel();
const childTable = await childColumn.getModel();
await childTable.getColumns();
await parentTable.getColumns();
const childTn = this.getTnPath(childTable);
const parentTn = this.getTnPath(parentTable);
switch (colOptions.type) {
case 'mm':
{
const vChildCol = await colOptions.getMMChildColumn();
const vParentCol = await colOptions.getMMParentColumn();
const vTable = await colOptions.getMMModel();
const vTn = this.getTnPath(vTable);
execQueries.push((trx, qb) =>
this.dbDriver(vTn)
.where({
[vChildCol.column_name]: this.dbDriver(childTn)
.select(childColumn.column_name)
.first(),
})
.delete(),
);
}
break;
case 'hm':
{
// skip if it's an mm table column
const relatedTable = await colOptions.getRelatedTable();
if (relatedTable.mm) {
break;
}
const childColumn = await Column.get({
colId: colOptions.fk_child_column_id,
});
execQueries.push((trx, qb) =>
trx(childTn)
.where({
[childColumn.column_name]: this.dbDriver.from(
qb
.select(parentColumn.column_name)
// .where(_wherePk(parentTable.primaryKeys, rowId))
.first()
.as('___cn_alias'),
),
})
.update({
[childColumn.column_name]: null,
}),
);
}
break;
}
}
const base = await Base.get(this.model.base_id);
trx = await this.dbDriver.transaction();
// unlink LTAR data
if (base.is_meta) {
for (const execQuery of execQueries) {
await execQuery(trx, qb.clone());
}
}
const deleteQb = qb.clone().transacting(trx).del();
const count = (await deleteQb) as any;
await trx.commit();
await this.afterBulkDelete(count, this.dbDriver, cookie, true); await this.afterBulkDelete(count, this.dbDriver, cookie, true);
return count; return count;
} catch (e) { } catch (e) {
if (trx) await trx.rollback();
throw e; throw e;
} }
} }

14
packages/nocodb/src/db/sql-client/lib/sqlite/SqliteClient.ts

@ -22,7 +22,8 @@ class SqliteClient extends KnexClient {
// sqlite does not support inserting default values and knex fires a warning without this flag // sqlite does not support inserting default values and knex fires a warning without this flag
connectionConfig.connection.useNullAsDefault = true; connectionConfig.connection.useNullAsDefault = true;
super(connectionConfig); super(connectionConfig);
this.sqlClient = knex(connectionConfig.connection); this.sqlClient =
connectionConfig?.knex || knex(connectionConfig.connection);
this.queries = queries; this.queries = queries;
this._version = {}; this._version = {};
} }
@ -1557,7 +1558,13 @@ class SqliteClient extends KnexClient {
upQuery, upQuery,
); );
await this.sqlClient.raw('PRAGMA foreign_keys = OFF;'); const fkCheckEnabled = (
await this.sqlClient.raw('PRAGMA foreign_keys;')
)?.[0]?.foreign_keys;
if (fkCheckEnabled)
await this.sqlClient.raw('PRAGMA foreign_keys = OFF;');
await this.sqlClient.raw('PRAGMA legacy_alter_table = ON;'); await this.sqlClient.raw('PRAGMA legacy_alter_table = ON;');
const trx = await this.sqlClient.transaction(); const trx = await this.sqlClient.transaction();
@ -1594,7 +1601,8 @@ class SqliteClient extends KnexClient {
log.ppe(e, _func); log.ppe(e, _func);
throw e; throw e;
} finally { } finally {
await this.sqlClient.raw('PRAGMA foreign_keys = ON;'); if (fkCheckEnabled)
await this.sqlClient.raw('PRAGMA foreign_keys = ON;');
await this.sqlClient.raw('PRAGMA legacy_alter_table = OFF;'); await this.sqlClient.raw('PRAGMA legacy_alter_table = OFF;');
} }

23
packages/nocodb/src/db/sql-mgr/v2/ProjectMgrv2.ts

@ -1,18 +1,21 @@
import SqlMgrv2 from './SqlMgrv2'; import { MetaService } from '../../../meta/meta.service'
import SqlMgrv2Trans from './SqlMgrv2Trans'; import SqlMgrv2 from './SqlMgrv2'
import SqlMgrv2Trans from './SqlMgrv2Trans'
// import type NcMetaIO from '../../../meta/NcMetaIO'; // import type NcMetaIO from '../../../meta/NcMetaIO';
import type Base from '../../../models/Base'; import type Base from '../../../models/Base'
export default class ProjectMgrv2 { export default class ProjectMgrv2 {
private static sqlMgrMap: { private static sqlMgrMap: {
[key: string]: SqlMgrv2; [key: string]: SqlMgrv2;
} = {}; } = {}
public static getSqlMgr(project: { id: string }, ncMeta: MetaService = null): SqlMgrv2 {
if (ncMeta) return new SqlMgrv2(project, ncMeta)
public static getSqlMgr(project: { id: string }): SqlMgrv2 {
if (!this.sqlMgrMap[project.id]) { if (!this.sqlMgrMap[project.id]) {
this.sqlMgrMap[project.id] = new SqlMgrv2(project); this.sqlMgrMap[project.id] = new SqlMgrv2(project)
} }
return this.sqlMgrMap[project.id]; return this.sqlMgrMap[project.id]
} }
public static async getSqlMgrTrans( public static async getSqlMgrTrans(
@ -21,8 +24,8 @@ export default class ProjectMgrv2 {
ncMeta: any, ncMeta: any,
base: Base, base: Base,
): Promise<SqlMgrv2Trans> { ): Promise<SqlMgrv2Trans> {
const sqlMgr = new SqlMgrv2Trans(project, ncMeta, base); const sqlMgr = new SqlMgrv2Trans(project, ncMeta, base)
await sqlMgr.startTransaction(base); await sqlMgr.startTransaction(base)
return sqlMgr; return sqlMgr
} }
} }

12
packages/nocodb/src/db/sql-mgr/v2/SqlMgrv2.ts

@ -5,12 +5,14 @@ import NcConnectionMgrv2 from '../../../utils/common/NcConnectionMgrv2';
import SqlClientFactory from '../../sql-client/lib/SqlClientFactory'; import SqlClientFactory from '../../sql-client/lib/SqlClientFactory';
import KnexMigratorv2 from '../../sql-migrator/lib/KnexMigratorv2'; import KnexMigratorv2 from '../../sql-migrator/lib/KnexMigratorv2';
import Debug from '../../util/Debug'; import Debug from '../../util/Debug';
import type { MetaService } from '../../../meta/meta.service';
import type Base from '../../../models/Base'; import type Base from '../../../models/Base';
const log = new Debug('SqlMgr'); const log = new Debug('SqlMgr');
export default class SqlMgrv2 { export default class SqlMgrv2 {
protected _migrator: KnexMigratorv2; protected _migrator: KnexMigratorv2;
protected ncMeta?: MetaService;
// @ts-ignore // @ts-ignore
private currentProjectFolder: any; private currentProjectFolder: any;
@ -20,18 +22,18 @@ export default class SqlMgrv2 {
* @param {String} args.toolDbPath - path to sqlite file that sql mgr will use * @param {String} args.toolDbPath - path to sqlite file that sql mgr will use
* @memberof SqlMgr * @memberof SqlMgr
*/ */
constructor(args: { id: string }) { constructor(args: { id: string }, ncMeta = null) {
const func = 'constructor'; const func = 'constructor';
log.api(`${func}:args:`, args); log.api(`${func}:args:`, args);
// this.metaDb = args.metaDb; // this.metaDb = args.metaDb;
this._migrator = new KnexMigratorv2(args); this._migrator = new KnexMigratorv2(args);
this.ncMeta = ncMeta;
return this;
} }
public async migrator(_base: Base) { public async migrator(_base: Base) {
return this._migrator; return this._migrator;
} }
public static async testConnection(args = {}) { public static async testConnection(args = {}) {
const client = await SqlClientFactory.create(args); const client = await SqlClientFactory.create(args);
return client.testConnection(); return client.testConnection();
@ -119,6 +121,10 @@ export default class SqlMgrv2 {
} }
protected async getSqlClient(base: Base) { protected async getSqlClient(base: Base) {
if (base.is_meta && this.ncMeta) {
return NcConnectionMgrv2.getSqlClient(base, this.ncMeta.knex);
}
return NcConnectionMgrv2.getSqlClient(base); return NcConnectionMgrv2.getSqlClient(base);
} }
} }

1
packages/nocodb/src/meta/meta.service.ts

@ -532,7 +532,6 @@ export class MetaService {
} else { } else {
query.where(idOrCondition); query.where(idOrCondition);
} }
return query.first(); return query.first();
} }

1
packages/nocodb/src/middlewares/extract-project-id/extract-project-id.middleware.ts

@ -37,6 +37,7 @@ export class ExtractProjectIdMiddleware implements NestMiddleware, CanActivate {
// extract project id based on request path params // extract project id based on request path params
if (params.projectName) { if (params.projectName) {
const project = await Project.getByTitleOrId(params.projectName); const project = await Project.getByTitleOrId(params.projectName);
if (!project) NcError.notFound('Project not found');
req.ncProjectId = project.id; req.ncProjectId = project.id;
res.locals.project = project; res.locals.project = project;
} }

16
packages/nocodb/src/models/Column.ts

@ -65,10 +65,13 @@ export default class Column<T = any> implements ColumnType {
Object.assign(this, data); Object.assign(this, data);
} }
public async getModel(): Promise<Model> { public async getModel(ncMeta = Noco.ncMeta): Promise<Model> {
return Model.getByIdOrName({ return Model.getByIdOrName(
id: this.fk_model_id, {
}); id: this.fk_model_id,
},
ncMeta,
);
} }
public static async insert<T>( public static async insert<T>(
@ -598,6 +601,11 @@ export default class Column<T = any> implements ColumnType {
static async delete(id, ncMeta = Noco.ncMeta) { static async delete(id, ncMeta = Noco.ncMeta) {
const col = await this.get({ colId: id }, ncMeta); const col = await this.get({ colId: id }, ncMeta);
// if column is not found, return
if (!col) {
return;
}
// todo: or instead of delete reset related foreign key value to null and handle in BaseModel // todo: or instead of delete reset related foreign key value to null and handle in BaseModel
// get qr code columns and delete // get qr code columns and delete

6
packages/nocodb/src/models/Model.ts

@ -368,10 +368,10 @@ export default class Model implements TableType {
} }
async delete(ncMeta = Noco.ncMeta, force = false): Promise<boolean> { async delete(ncMeta = Noco.ncMeta, force = false): Promise<boolean> {
await Audit.deleteRowComments(this.id); await Audit.deleteRowComments(this.id, ncMeta);
for (const view of await this.getViews(true)) { for (const view of await this.getViews(true, ncMeta)) {
await view.delete(); await view.delete(ncMeta);
} }
for (const col of await this.getColumns(ncMeta)) { for (const col of await this.getColumns(ncMeta)) {

7
packages/nocodb/src/models/Project.ts

@ -10,7 +10,7 @@ import NocoCache from '../cache/NocoCache';
import Base from './/Base'; import Base from './/Base';
import { ProjectUser } from './index'; import { ProjectUser } from './index';
import type { BoolType, MetaType, ProjectType } from 'nocodb-sdk'; import type { BoolType, MetaType, ProjectType } from 'nocodb-sdk';
import type { DB_TYPES } from './/Base'; import type { DB_TYPES } from './Base';
export default class Project implements ProjectType { export default class Project implements ProjectType {
public id: string; public id: string;
@ -309,6 +309,11 @@ export default class Project implements ProjectType {
`${CacheScope.PROJECT}:${projectId}`, `${CacheScope.PROJECT}:${projectId}`,
CacheDelDirection.CHILD_TO_PARENT, CacheDelDirection.CHILD_TO_PARENT,
); );
await ncMeta.metaDelete(null, null, MetaTable.AUDIT, {
project_id: projectId,
});
return await ncMeta.metaDelete(null, null, MetaTable.PROJECT, projectId); return await ncMeta.metaDelete(null, null, MetaTable.PROJECT, projectId);
} }

10
packages/nocodb/src/models/View.ts

@ -1001,9 +1001,9 @@ export default class View implements ViewType {
// @ts-ignore // @ts-ignore
static async delete(viewId, ncMeta = Noco.ncMeta) { static async delete(viewId, ncMeta = Noco.ncMeta) {
const view = await this.get(viewId); const view = await this.get(viewId, ncMeta);
await Sort.deleteAll(viewId); await Sort.deleteAll(viewId, ncMeta);
await Filter.deleteAll(viewId); await Filter.deleteAll(viewId, ncMeta);
const table = this.extractViewTableName(view); const table = this.extractViewTableName(view);
const tableScope = this.extractViewTableNameScope(view); const tableScope = this.extractViewTableNameScope(view);
const columnTable = this.extractViewColumnsTableName(view); const columnTable = this.extractViewColumnsTableName(view);
@ -1273,8 +1273,8 @@ export default class View implements ViewType {
); );
} }
async delete() { async delete(ncMeta = Noco.ncMeta){
await View.delete(this.id); await View.delete(this.id, ncMeta);
} }
static async shareViewList(tableId, ncMeta = Noco.ncMeta) { static async shareViewList(tableId, ncMeta = Noco.ncMeta) {

2
packages/nocodb/src/modules/global/init-meta-service.provider.ts

@ -26,7 +26,7 @@ export const InitMetaServiceProvider: Provider = {
const config = await NcConfig.createByEnv(); const config = await NcConfig.createByEnv();
// set version // set version
process.env.NC_VERSION = '0107004'; process.env.NC_VERSION = '0108002';
// init cache // init cache
await NocoCache.init(); await NocoCache.init();

215
packages/nocodb/src/services/columns.service.ts

@ -38,8 +38,8 @@ import {
import Noco from '../Noco'; import Noco from '../Noco';
import NcConnectionMgrv2 from '../utils/common/NcConnectionMgrv2'; import NcConnectionMgrv2 from '../utils/common/NcConnectionMgrv2';
import { MetaTable } from '../utils/globals'; import { MetaTable } from '../utils/globals';
import { MetaService } from '../meta/meta.service';
import type { LinkToAnotherRecordColumn, Project } from '../models'; import type { LinkToAnotherRecordColumn, Project } from '../models';
import type { MetaService } from '../meta/meta.service';
import type SqlMgrv2 from '../db/sql-mgr/v2/SqlMgrv2'; import type SqlMgrv2 from '../db/sql-mgr/v2/SqlMgrv2';
import type { import type {
ColumnReqType, ColumnReqType,
@ -57,6 +57,8 @@ export enum Altered {
@Injectable() @Injectable()
export class ColumnsService { export class ColumnsService {
constructor(private readonly metaService: MetaService) {}
async columnUpdate(param: { async columnUpdate(param: {
req?: any; req?: any;
columnId: string; columnId: string;
@ -1142,21 +1144,23 @@ export class ColumnsService {
return table; return table;
} }
async columnDelete(param: { req?: any; columnId: string }) { async columnDelete(
const column = await Column.get({ colId: param.columnId }); param: { req?: any; columnId: string },
const table = await Model.getWithInfo({ ncMeta = this.metaService,
id: column.fk_model_id, ) {
}); const column = await Column.get({ colId: param.columnId }, ncMeta);
const base = await Base.get(table.base_id); const table = await Model.getWithInfo(
{
// const ncMeta = await Noco.ncMeta.startTransaction(); id: column.fk_model_id,
// const sql-mgr = await ProjectMgrv2.getSqlMgrTrans( },
// { id: base.project_id }, ncMeta,
// ncMeta, );
// base const base = await Base.get(table.base_id, ncMeta);
// );
const sqlMgr = await ProjectMgrv2.getSqlMgr({ id: base.project_id }); const sqlMgr = await ProjectMgrv2.getSqlMgr(
{ id: base.project_id },
ncMeta,
);
switch (column.uidt) { switch (column.uidt) {
case UITypes.Lookup: case UITypes.Lookup:
@ -1164,17 +1168,17 @@ export class ColumnsService {
case UITypes.QrCode: case UITypes.QrCode:
case UITypes.Barcode: case UITypes.Barcode:
case UITypes.Formula: case UITypes.Formula:
await Column.delete(param.columnId); await Column.delete(param.columnId, ncMeta);
break; break;
case UITypes.LinkToAnotherRecord: case UITypes.LinkToAnotherRecord:
{ {
const relationColOpt = const relationColOpt =
await column.getColOptions<LinkToAnotherRecordColumn>(); await column.getColOptions<LinkToAnotherRecordColumn>(ncMeta);
const childColumn = await relationColOpt.getChildColumn(); const childColumn = await relationColOpt.getChildColumn(ncMeta);
const childTable = await childColumn.getModel(); const childTable = await childColumn.getModel(ncMeta);
const parentColumn = await relationColOpt.getParentColumn(); const parentColumn = await relationColOpt.getParentColumn(ncMeta);
const parentTable = await parentColumn.getModel(); const parentTable = await parentColumn.getModel(ncMeta);
switch (relationColOpt.type) { switch (relationColOpt.type) {
case 'bt': case 'bt':
@ -1188,15 +1192,19 @@ export class ColumnsService {
parentColumn, parentColumn,
parentTable, parentTable,
sqlMgr, sqlMgr,
// ncMeta ncMeta,
}); });
} }
break; break;
case 'mm': case 'mm':
{ {
const mmTable = await relationColOpt.getMMModel(); const mmTable = await relationColOpt.getMMModel(ncMeta);
const mmParentCol = await relationColOpt.getMMParentColumn(); const mmParentCol = await relationColOpt.getMMParentColumn(
const mmChildCol = await relationColOpt.getMMChildColumn(); ncMeta,
);
const mmChildCol = await relationColOpt.getMMChildColumn(
ncMeta,
);
await this.deleteHmOrBtRelation( await this.deleteHmOrBtRelation(
{ {
@ -1207,7 +1215,7 @@ export class ColumnsService {
parentTable: parentTable, parentTable: parentTable,
childColumn: mmParentCol, childColumn: mmParentCol,
base, base,
// ncMeta ncMeta,
}, },
true, true,
); );
@ -1221,18 +1229,18 @@ export class ColumnsService {
parentTable: childTable, parentTable: childTable,
childColumn: mmChildCol, childColumn: mmChildCol,
base, base,
// ncMeta ncMeta,
}, },
true, true,
); );
const columnsInRelatedTable: Column[] = await relationColOpt const columnsInRelatedTable: Column[] = await relationColOpt
.getRelatedTable() .getRelatedTable(ncMeta)
.then((m) => m.getColumns()); .then((m) => m.getColumns(ncMeta));
for (const c of columnsInRelatedTable) { for (const c of columnsInRelatedTable) {
if (c.uidt !== UITypes.LinkToAnotherRecord) continue; if (c.uidt !== UITypes.LinkToAnotherRecord) continue;
const colOpt = const colOpt =
await c.getColOptions<LinkToAnotherRecordColumn>(); await c.getColOptions<LinkToAnotherRecordColumn>(ncMeta);
if ( if (
colOpt.type === 'mm' && colOpt.type === 'mm' &&
colOpt.fk_parent_column_id === childColumn.id && colOpt.fk_parent_column_id === childColumn.id &&
@ -1241,55 +1249,55 @@ export class ColumnsService {
colOpt.fk_mm_parent_column_id === mmChildCol.id && colOpt.fk_mm_parent_column_id === mmChildCol.id &&
colOpt.fk_mm_child_column_id === mmParentCol.id colOpt.fk_mm_child_column_id === mmParentCol.id
) { ) {
await Column.delete(c.id); await Column.delete(c.id, ncMeta);
break; break;
} }
} }
await Column.delete(relationColOpt.fk_column_id); await Column.delete(relationColOpt.fk_column_id, ncMeta);
// delete bt columns in m2m table // delete bt columns in m2m table
await mmTable.getColumns(); await mmTable.getColumns(ncMeta);
for (const c of mmTable.columns) { for (const c of mmTable.columns) {
if (c.uidt !== UITypes.LinkToAnotherRecord) continue; if (c.uidt !== UITypes.LinkToAnotherRecord) continue;
const colOpt = const colOpt =
await c.getColOptions<LinkToAnotherRecordColumn>(); await c.getColOptions<LinkToAnotherRecordColumn>(ncMeta);
if (colOpt.type === 'bt') { if (colOpt.type === 'bt') {
await Column.delete(c.id); await Column.delete(c.id, ncMeta);
} }
} }
// delete hm columns in parent table // delete hm columns in parent table
await parentTable.getColumns(); await parentTable.getColumns(ncMeta);
for (const c of parentTable.columns) { for (const c of parentTable.columns) {
if (c.uidt !== UITypes.LinkToAnotherRecord) continue; if (c.uidt !== UITypes.LinkToAnotherRecord) continue;
const colOpt = const colOpt =
await c.getColOptions<LinkToAnotherRecordColumn>(); await c.getColOptions<LinkToAnotherRecordColumn>(ncMeta);
if (colOpt.fk_related_model_id === mmTable.id) { if (colOpt.fk_related_model_id === mmTable.id) {
await Column.delete(c.id); await Column.delete(c.id, ncMeta);
} }
} }
// delete hm columns in child table // delete hm columns in child table
await childTable.getColumns(); await childTable.getColumns(ncMeta);
for (const c of childTable.columns) { for (const c of childTable.columns) {
if (c.uidt !== UITypes.LinkToAnotherRecord) continue; if (c.uidt !== UITypes.LinkToAnotherRecord) continue;
const colOpt = const colOpt =
await c.getColOptions<LinkToAnotherRecordColumn>(); await c.getColOptions<LinkToAnotherRecordColumn>(ncMeta);
if (colOpt.fk_related_model_id === mmTable.id) { if (colOpt.fk_related_model_id === mmTable.id) {
await Column.delete(c.id); await Column.delete(c.id, ncMeta);
} }
} }
// retrieve columns in m2m table again // retrieve columns in m2m table again
await mmTable.getColumns(); await mmTable.getColumns(ncMeta);
// ignore deleting table if it has more than 2 columns // ignore deleting table if it has more than 2 columns
// the expected 2 columns would be table1_id & table2_id // the expected 2 columns would be table1_id & table2_id
if (mmTable.columns.length === 2) { if (mmTable.columns.length === 2) {
(mmTable as any).tn = mmTable.table_name; (mmTable as any).tn = mmTable.table_name;
await sqlMgr.sqlOpPlus(base, 'tableDelete', mmTable); await sqlMgr.sqlOpPlus(base, 'tableDelete', mmTable);
await mmTable.delete(); await mmTable.delete(ncMeta);
} }
} }
break; break;
@ -1337,26 +1345,30 @@ export class ColumnsService {
await sqlMgr.sqlOpPlus(base, 'tableUpdate', tableUpdateBody); await sqlMgr.sqlOpPlus(base, 'tableUpdate', tableUpdateBody);
await Column.delete(param.columnId); await Column.delete(param.columnId, ncMeta);
} }
} }
await Audit.insert({ await Audit.insert(
project_id: base.project_id, {
op_type: AuditOperationTypes.TABLE_COLUMN, project_id: base.project_id,
op_sub_type: AuditOperationSubTypes.DELETE, op_type: AuditOperationTypes.TABLE_COLUMN,
user: param?.req?.user?.email, op_sub_type: AuditOperationSubTypes.DELETE,
description: `The column ${column.column_name} with alias ${column.title} from table ${table.table_name} has been deleted`, user: param?.req?.user?.email,
ip: param?.req.clientIp, description: `The column ${column.column_name} with alias ${column.title} from table ${table.table_name} has been deleted`,
}); ip: param?.req.clientIp,
},
await table.getColumns(); ncMeta,
);
await table.getColumns(ncMeta);
const displayValueColumn = mapDefaultDisplayValue(table.columns); const displayValueColumn = mapDefaultDisplayValue(table.columns);
if (displayValueColumn) { if (displayValueColumn) {
await Model.updatePrimaryColumn( await Model.updatePrimaryColumn(
displayValueColumn.fk_model_id, displayValueColumn.fk_model_id,
displayValueColumn.id, displayValueColumn.id,
ncMeta,
); );
} }
@ -1394,45 +1406,47 @@ export class ColumnsService {
if (!relationColOpt) { if (!relationColOpt) {
foreignKeyName = ( foreignKeyName = (
( (
await childTable.getColumns().then((cols) => { await childTable.getColumns(ncMeta).then(async (cols) => {
return cols?.find((c) => { for (const col of cols) {
return ( if (col.uidt === UITypes.LinkToAnotherRecord) {
c.uidt === UITypes.LinkToAnotherRecord && const colOptions =
c.colOptions.fk_related_model_id === parentTable.id && await col.getColOptions<LinkToAnotherRecordColumn>(ncMeta);
(c.colOptions as LinkToAnotherRecordType).fk_child_column_id === console.log(colOptions);
childColumn.id && if (colOptions.fk_related_model_id === parentTable.id) {
(c.colOptions as LinkToAnotherRecordType) return { colOptions };
.fk_parent_column_id === parentColumn.id }
); }
}); }
}) })
).colOptions as LinkToAnotherRecordType )?.colOptions as LinkToAnotherRecordType
).fk_index_name; ).fk_index_name;
} else { } else {
foreignKeyName = relationColOpt.fk_index_name; foreignKeyName = relationColOpt.fk_index_name;
} }
// todo: handle relation delete exception if (!relationColOpt?.virtual) {
try { // todo: handle relation delete exception
await sqlMgr.sqlOpPlus(base, 'relationDelete', { try {
childColumn: childColumn.column_name, await sqlMgr.sqlOpPlus(base, 'relationDelete', {
childTable: childTable.table_name, childColumn: childColumn.column_name,
parentTable: parentTable.table_name, childTable: childTable.table_name,
parentColumn: parentColumn.column_name, parentTable: parentTable.table_name,
foreignKeyName, parentColumn: parentColumn.column_name,
}); foreignKeyName,
} catch (e) { });
console.log(e); } catch (e) {
console.log(e);
}
} }
if (!relationColOpt) return; if (!relationColOpt) return;
const columnsInRelatedTable: Column[] = await relationColOpt const columnsInRelatedTable: Column[] = await relationColOpt
.getRelatedTable() .getRelatedTable(ncMeta)
.then((m) => m.getColumns()); .then((m) => m.getColumns(ncMeta));
const relType = relationColOpt.type === 'bt' ? 'hm' : 'bt'; const relType = relationColOpt.type === 'bt' ? 'hm' : 'bt';
for (const c of columnsInRelatedTable) { for (const c of columnsInRelatedTable) {
if (c.uidt !== UITypes.LinkToAnotherRecord) continue; if (c.uidt !== UITypes.LinkToAnotherRecord) continue;
const colOpt = await c.getColOptions<LinkToAnotherRecordColumn>(); const colOpt = await c.getColOptions<LinkToAnotherRecordColumn>(ncMeta);
if ( if (
colOpt.fk_parent_column_id === parentColumn.id && colOpt.fk_parent_column_id === parentColumn.id &&
colOpt.fk_child_column_id === childColumn.id && colOpt.fk_child_column_id === childColumn.id &&
@ -1447,9 +1461,34 @@ export class ColumnsService {
await Column.delete(relationColOpt.fk_column_id, ncMeta); await Column.delete(relationColOpt.fk_column_id, ncMeta);
if (!ignoreFkDelete) { if (!ignoreFkDelete) {
const cTable = await Model.getWithInfo({ const cTable = await Model.getWithInfo(
id: childTable.id, {
}); id: childTable.id,
},
ncMeta,
);
// if virtual column delete all index before deleting the column
if (relationColOpt?.virtual) {
const indexes =
(
await sqlMgr.sqlOp(base, 'indexList', {
tn: cTable.table_name,
})
)?.data?.list ?? [];
for (const index of indexes) {
if (index.cn !== childColumn.column_name) continue;
await sqlMgr.sqlOpPlus(base, 'indexDelete', {
...index,
tn: cTable.table_name,
columns: [childColumn.column_name],
indexName: index.index_name,
});
}
}
const tableUpdateBody = { const tableUpdateBody = {
...cTable, ...cTable,
tn: cTable.table_name, tn: cTable.table_name,
@ -1499,6 +1538,12 @@ export class ColumnsService {
const sqlMgr = await ProjectMgrv2.getSqlMgr({ const sqlMgr = await ProjectMgrv2.getSqlMgr({
id: param.base.project_id, id: param.base.project_id,
}); });
// if xcdb base then treat as virtual relation to avoid creating foreign key
if (param.base.is_meta) {
(param.column as LinkToAnotherColumnReqType).virtual = true;
}
if ( if (
(param.column as LinkToAnotherColumnReqType).type === 'hm' || (param.column as LinkToAnotherColumnReqType).type === 'hm' ||
(param.column as LinkToAnotherColumnReqType).type === 'bt' (param.column as LinkToAnotherColumnReqType).type === 'bt'
@ -1570,7 +1615,7 @@ export class ColumnsService {
} }
// todo: create index for virtual relations as well // todo: create index for virtual relations as well
// create index for foreign key in pg // create index for foreign key in pg
if ( if (
param.base.type === 'pg' || param.base.type === 'pg' ||
(param.column as LinkToAnotherColumnReqType).virtual (param.column as LinkToAnotherColumnReqType).virtual
@ -1729,6 +1774,7 @@ export class ColumnsService {
fk_mm_child_column_id: childCol.id, fk_mm_child_column_id: childCol.id,
fk_mm_parent_column_id: parentCol.id, fk_mm_parent_column_id: parentCol.id,
fk_related_model_id: parent.id, fk_related_model_id: parent.id,
virtual: (param.column as LinkToAnotherColumnReqType).virtual,
}); });
await Column.insert({ await Column.insert({
title: getUniqueColumnAliasName( title: getUniqueColumnAliasName(
@ -1748,6 +1794,7 @@ export class ColumnsService {
fk_mm_child_column_id: parentCol.id, fk_mm_child_column_id: parentCol.id,
fk_mm_parent_column_id: childCol.id, fk_mm_parent_column_id: childCol.id,
fk_related_model_id: child.id, fk_related_model_id: child.id,
virtual: (param.column as LinkToAnotherColumnReqType).virtual,
}); });
// todo: create index for virtual relations as well // todo: create index for virtual relations as well

12
packages/nocodb/src/services/datas.service.ts

@ -103,11 +103,15 @@ export class DatasService {
dbDriver: await NcConnectionMgrv2.get(base), dbDriver: await NcConnectionMgrv2.get(base),
}); });
// todo: Should have error http status code // if xcdb project skip checking for LTAR
const message = await baseModel.hasLTARData(param.rowId, model); if (!base.is_meta) {
if (message.length) { // todo: Should have error http status code
NcError.badRequest(message); const message = await baseModel.hasLTARData(param.rowId, model);
if (message.length) {
NcError.badRequest(message);
}
} }
return await baseModel.delByPk(param.rowId, null, param.cookie); return await baseModel.delByPk(param.rowId, null, param.cookie);
} }

105
packages/nocodb/src/services/tables.service.ts

@ -14,9 +14,19 @@ import getColumnPropsFromUIDT from '../helpers/getColumnPropsFromUIDT';
import getColumnUiType from '../helpers/getColumnUiType'; import getColumnUiType from '../helpers/getColumnUiType';
import getTableNameAlias, { getColumnNameAlias } from '../helpers/getTableName'; import getTableNameAlias, { getColumnNameAlias } from '../helpers/getTableName';
import mapDefaultDisplayValue from '../helpers/mapDefaultDisplayValue'; import mapDefaultDisplayValue from '../helpers/mapDefaultDisplayValue';
import { Audit, Column, Model, ModelRoleVisibility, Project } from '../models'; import {
Audit,
Base,
Column,
Model,
ModelRoleVisibility,
Project,
} from '../models';
import Noco from '../Noco';
import NcConnectionMgrv2 from '../utils/common/NcConnectionMgrv2'; import NcConnectionMgrv2 from '../utils/common/NcConnectionMgrv2';
import { validatePayload } from '../helpers'; import { validatePayload } from '../helpers';
import { ColumnsService } from './columns.service';
import type { MetaService } from '../meta/meta.service';
import type { LinkToAnotherRecordColumn, User, View } from '../models'; import type { LinkToAnotherRecordColumn, User, View } from '../models';
import type { import type {
ColumnType, ColumnType,
@ -26,6 +36,8 @@ import type {
@Injectable() @Injectable()
export class TablesService { export class TablesService {
constructor(private readonly columnsService: ColumnsService) {}
async tableUpdate(param: { async tableUpdate(param: {
tableId: any; tableId: any;
table: TableReqType & { project_id?: string }; table: TableReqType & { project_id?: string };
@ -142,11 +154,14 @@ export class TablesService {
const table = await Model.getByIdOrName({ id: param.tableId }); const table = await Model.getByIdOrName({ id: param.tableId });
await table.getColumns(); await table.getColumns();
const project = await Project.getWithInfo(table.project_id);
const base = project.bases.find((b) => b.id === table.base_id);
const relationColumns = table.columns.filter( const relationColumns = table.columns.filter(
(c) => c.uidt === UITypes.LinkToAnotherRecord, (c) => c.uidt === UITypes.LinkToAnotherRecord,
); );
if (relationColumns?.length) { if (relationColumns?.length && !base.is_meta) {
const referredTables = await Promise.all( const referredTables = await Promise.all(
relationColumns.map(async (c) => relationColumns.map(async (c) =>
c c
@ -162,37 +177,69 @@ export class TablesService {
); );
} }
const project = await Project.getWithInfo(table.project_id); // start a transaction
const base = project.bases.find((b) => b.id === table.base_id); const ncMeta = await (Noco.ncMeta as MetaService).startTransaction();
const sqlMgr = await ProjectMgrv2.getSqlMgr(project); let result;
(table as any).tn = table.table_name; try {
table.columns = table.columns.filter((c) => !isVirtualCol(c)); // delete all relations
table.columns.forEach((c) => { for (const c of relationColumns) {
(c as any).cn = c.column_name; // skip if column is hasmany relation to mm table
}); if (c.system) {
continue;
}
// verify column exist or not and based on that delete the column
if (!(await Column.get({ colId: c.id }, ncMeta))) {
continue;
}
await this.columnsService.columnDelete(
{
req: param.req,
columnId: c.id,
},
ncMeta,
);
}
if (table.type === ModelTypes.TABLE) { const sqlMgr = await ProjectMgrv2.getSqlMgr(project, ncMeta);
await sqlMgr.sqlOpPlus(base, 'tableDelete', table); (table as any).tn = table.table_name;
} else if (table.type === ModelTypes.VIEW) { table.columns = table.columns.filter((c) => !isVirtualCol(c));
await sqlMgr.sqlOpPlus(base, 'viewDelete', { table.columns.forEach((c) => {
...table, (c as any).cn = c.column_name;
view_name: table.table_name,
}); });
}
await Audit.insert({ if (table.type === ModelTypes.TABLE) {
project_id: project.id, await sqlMgr.sqlOpPlus(base, 'tableDelete', table);
base_id: base.id, } else if (table.type === ModelTypes.VIEW) {
op_type: AuditOperationTypes.TABLE, await sqlMgr.sqlOpPlus(base, 'viewDelete', {
op_sub_type: AuditOperationSubTypes.DELETE, ...table,
user: param.user?.email, view_name: table.table_name,
description: `Deleted ${table.type} ${table.table_name} with alias ${table.title} `, });
ip: param.req?.clientIp, }
}).then(() => {});
T.emit('evt', { evt_type: 'table:deleted' });
return table.delete(); await Audit.insert(
{
project_id: project.id,
base_id: base.id,
op_type: AuditOperationTypes.TABLE,
op_sub_type: AuditOperationSubTypes.DELETE,
user: param.user?.email,
description: `Deleted ${table.type} ${table.table_name} with alias ${table.title} `,
ip: param.req?.clientIp,
},
ncMeta,
).then(() => {});
T.emit('evt', { evt_type: 'table:deleted' });
result = await table.delete(ncMeta);
await ncMeta.commit();
} catch (e) {
await ncMeta.rollback();
throw e;
}
return result;
} }
async getTableWithAccessibleViews(param: { tableId: string; user: User }) { async getTableWithAccessibleViews(param: { tableId: string; user: User }) {

2
packages/nocodb/src/version-upgrader/NcUpgrader.ts

@ -14,6 +14,7 @@ import ncProjectEnvUpgrader0011045 from './ncProjectEnvUpgrader0011045';
import ncProjectEnvUpgrader from './ncProjectEnvUpgrader'; import ncProjectEnvUpgrader from './ncProjectEnvUpgrader';
import ncHookUpgrader from './ncHookUpgrader'; import ncHookUpgrader from './ncHookUpgrader';
import ncProjectConfigUpgrader from './ncProjectConfigUpgrader'; import ncProjectConfigUpgrader from './ncProjectConfigUpgrader';
import ncXcdbLTARUpgrader from './ncXcdbLTARUpgrader';
import type { MetaService } from '../meta/meta.service'; import type { MetaService } from '../meta/meta.service';
import type { NcConfig } from '../interface/config'; import type { NcConfig } from '../interface/config';
@ -50,6 +51,7 @@ export default class NcUpgrader {
{ name: '0105003', handler: ncFilterUpgrader_0105003 }, { name: '0105003', handler: ncFilterUpgrader_0105003 },
{ name: '0105004', handler: ncHookUpgrader }, { name: '0105004', handler: ncHookUpgrader },
{ name: '0107004', handler: ncProjectConfigUpgrader }, { name: '0107004', handler: ncProjectConfigUpgrader },
{ name: '0108002', handler: ncXcdbLTARUpgrader },
]; ];
if (!(await ctx.ncMeta.knexConnection?.schema?.hasTable?.('nc_store'))) { if (!(await ctx.ncMeta.knexConnection?.schema?.hasTable?.('nc_store'))) {
return; return;

384
packages/nocodb/src/version-upgrader/ncFilterUpgrader_0104004.ts

@ -1,359 +1,43 @@
import { UITypes } from 'nocodb-sdk'; import { OrgUserRoles } from 'nocodb-sdk';
import { NC_APP_SETTINGS } from '../constants';
import Store from '../models/Store';
import { MetaTable } from '../utils/globals'; import { MetaTable } from '../utils/globals';
import Column from '../models/Column';
import Filter from '../models/Filter';
import Project from '../models/Project';
import type { MetaService } from '../meta/meta.service';
import type { NcUpgraderCtx } from './NcUpgrader'; import type { NcUpgraderCtx } from './NcUpgrader';
import type { SelectOptionsType } from 'nocodb-sdk';
// as of 0.104.3, almost all filter operators are available to all column types /** Upgrader for upgrading roles */
// while some of them aren't supposed to be shown export default async function ({ ncMeta }: NcUpgraderCtx) {
// this upgrader is to remove those unsupported filters / migrate to the correct filter const users = await ncMeta.metaList2(null, null, MetaTable.USERS);
// Change Summary: for (const user of users) {
// - Text-based columns: user.roles = user.roles
// - remove `>`, `<`, `>=`, `<=` .split(',')
// - Numeric-based / SingleSelect columns: .map((r) => {
// - remove `like` // update old role names with new roles
// - migrate `null`, and `empty` to `blank` if (r === 'user') {
// - Checkbox columns: return OrgUserRoles.CREATOR;
// - remove `equal` } else if (r === 'user-new') {
// - migrate `empty` and `null` to `notchecked` return OrgUserRoles.VIEWER;
// - MultiSelect columns: }
// - remove `like` return r;
// - migrate `equal`, `null`, `empty` })
// - Attachment columns: .join(',');
// - remove `>`, `<`, `>=`, `<=`, `equal` await ncMeta.metaUpdate(
// - migrate `empty`, `null` to `blank` null,
// - LTAR columns: null,
// - remove `>`, `<`, `>=`, `<=` MetaTable.USERS,
// - migrate `empty`, `null` to `blank` { roles: user.roles },
// - Lookup columns: user.id,
// - migrate `empty`, `null` to `blank`
// - Duration columns:
// - remove `like`
// - migrate `empty`, `null` to `blank`
const removeEqualFilters = (filter, ncMeta) => {
const actions = [];
// remove `is equal`, `is not equal`
if (['eq', 'neq'].includes(filter.comparison_op)) {
actions.push(Filter.delete(filter.id, ncMeta));
}
return actions;
};
const removeArithmeticFilters = (filter, ncMeta) => {
const actions = [];
// remove `>`, `<`, `>=`, `<=`
if (['gt', 'lt', 'gte', 'lte'].includes(filter.comparison_op)) {
actions.push(Filter.delete(filter.id, ncMeta));
}
return actions;
};
const removeLikeFilters = (filter, ncMeta) => {
const actions = [];
// remove `is like`, `is not like`
if (['like', 'nlike'].includes(filter.comparison_op)) {
actions.push(Filter.delete(filter.id, ncMeta));
}
return actions;
};
const migrateNullAndEmptyToBlankFilters = (filter, ncMeta) => {
const actions = [];
if (['empty', 'null'].includes(filter.comparison_op)) {
// migrate to blank
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'blank',
},
ncMeta,
),
);
} else if (['notempty', 'notnull'].includes(filter.comparison_op)) {
// migrate to not blank
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'notblank',
},
ncMeta,
),
);
}
return actions;
};
const migrateMultiSelectEq = async (filter, col: Column, ncMeta) => {
// only allow eq / neq
if (!['eq', 'neq'].includes(filter.comparison_op)) return;
// if there is no value -> delete this filter
if (!filter.value) {
return await Filter.delete(filter.id, ncMeta);
}
// options inputted from users
const options = filter.value.split(',');
// retrieve the possible col options
const colOptions = (await col.getColOptions()) as SelectOptionsType;
// only include valid options as the input value becomes dropdown type now
const validOptions = [];
for (const option of options) {
if (colOptions.options.includes(option)) {
validOptions.push(option);
}
}
const newFilterValue = validOptions.join(',');
// if all inputted options are invalid -> delete this filter
if (!newFilterValue) {
return await Filter.delete(filter.id, ncMeta);
}
const actions = [];
if (filter.comparison_op === 'eq') {
// migrate to `contains all of`
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'anyof',
value: newFilterValue,
},
ncMeta,
),
);
} else if (filter.comparison_op === 'neq') {
// migrate to `doesn't contain all of`
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'nanyof',
value: newFilterValue,
},
ncMeta,
),
);
}
return await Promise.all(actions);
};
const migrateToCheckboxFilter = (filter, ncMeta) => {
const actions = [];
const possibleTrueValues = ['true', 'True', '1', 'T', 'Y'];
const possibleFalseValues = ['false', 'False', '0', 'F', 'N'];
if (['empty', 'null'].includes(filter.comparison_op)) {
// migrate to not checked
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'notchecked',
},
ncMeta,
),
);
} else if (['notempty', 'notnull'].includes(filter.comparison_op)) {
// migrate to checked
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'checked',
},
ncMeta,
),
); );
} else if (filter.comparison_op === 'eq') {
if (possibleTrueValues.includes(filter.value)) {
// migrate to checked
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'checked',
value: '',
},
ncMeta,
),
);
} else if (possibleFalseValues.includes(filter.value)) {
// migrate to notchecked
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'notchecked',
value: '',
},
ncMeta,
),
);
} else {
// invalid value - good to delete
actions.push(Filter.delete(filter.id, ncMeta));
}
} else if (filter.comparison_op === 'neq') {
if (possibleFalseValues.includes(filter.value)) {
// migrate to checked
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'checked',
value: '',
},
ncMeta,
),
);
} else if (possibleTrueValues.includes(filter.value)) {
// migrate to not checked
actions.push(
Filter.update(
filter.id,
{
comparison_op: 'notchecked',
value: '',
},
ncMeta,
),
);
} else {
// invalid value - good to delete
actions.push(Filter.delete(filter.id, ncMeta));
}
}
return actions;
};
async function migrateFilters(ncMeta: MetaService) {
const filters = await ncMeta.metaList2(null, null, MetaTable.FILTER_EXP);
for (const filter of filters) {
if (!filter.fk_column_id || filter.is_group) {
continue;
}
const col = await Column.get({ colId: filter.fk_column_id }, ncMeta);
if (
[
UITypes.SingleLineText,
UITypes.LongText,
UITypes.PhoneNumber,
UITypes.Email,
UITypes.URL,
].includes(col.uidt)
) {
await Promise.all(removeArithmeticFilters(filter, ncMeta));
} else if (
[
// numeric fields
UITypes.Duration,
UITypes.Currency,
UITypes.Percent,
UITypes.Number,
UITypes.Decimal,
UITypes.Rating,
UITypes.Rollup,
// select fields
UITypes.SingleSelect,
].includes(col.uidt)
) {
await Promise.all([
...removeLikeFilters(filter, ncMeta),
...migrateNullAndEmptyToBlankFilters(filter, ncMeta),
]);
} else if (col.uidt === UITypes.Checkbox) {
await Promise.all(migrateToCheckboxFilter(filter, ncMeta));
} else if (col.uidt === UITypes.MultiSelect) {
await Promise.all([
...removeLikeFilters(filter, ncMeta),
...migrateNullAndEmptyToBlankFilters(filter, ncMeta),
]);
await migrateMultiSelectEq(filter, col, ncMeta);
} else if (col.uidt === UITypes.Attachment) {
await Promise.all([
...removeArithmeticFilters(filter, ncMeta),
...removeEqualFilters(filter, ncMeta),
...migrateNullAndEmptyToBlankFilters(filter, ncMeta),
]);
} else if (col.uidt === UITypes.LinkToAnotherRecord) {
await Promise.all([
...removeArithmeticFilters(filter, ncMeta),
...migrateNullAndEmptyToBlankFilters(filter, ncMeta),
]);
} else if (col.uidt === UITypes.Lookup) {
await Promise.all([
...removeArithmeticFilters(filter, ncMeta),
...migrateNullAndEmptyToBlankFilters(filter, ncMeta),
]);
} else if (col.uidt === UITypes.Duration) {
await Promise.all([
...removeLikeFilters(filter, ncMeta),
...migrateNullAndEmptyToBlankFilters(filter, ncMeta),
]);
}
}
}
async function updateProjectMeta(ncMeta: MetaService) {
const projectHasEmptyOrFilters: Record<string, boolean> = {};
const filters = await ncMeta.metaList2(null, null, MetaTable.FILTER_EXP);
const actions = [];
for (const filter of filters) {
if (
['notempty', 'notnull', 'empty', 'null'].includes(filter.comparison_op)
) {
projectHasEmptyOrFilters[filter.project_id] = true;
}
} }
const projects = await ncMeta.metaList2(null, null, MetaTable.PROJECT); // set invite only signup if user have environment variable set
if (process.env.NC_INVITE_ONLY_SIGNUP) {
const defaultProjectMeta = { await Store.saveOrUpdate(
showNullAndEmptyInFilter: false, {
}; value: '{ "invite_only_signup": true }',
key: NC_APP_SETTINGS,
for (const project of projects) { },
const oldProjectMeta = project.meta; ncMeta,
let newProjectMeta = defaultProjectMeta;
try {
newProjectMeta =
(typeof oldProjectMeta === 'string'
? JSON.parse(oldProjectMeta)
: oldProjectMeta) ?? defaultProjectMeta;
} catch {}
newProjectMeta = {
...newProjectMeta,
showNullAndEmptyInFilter: projectHasEmptyOrFilters[project.id] ?? false,
};
actions.push(
Project.update(
project.id,
{
meta: JSON.stringify(newProjectMeta),
},
ncMeta,
),
); );
} }
await Promise.all(actions);
}
export default async function ({ ncMeta }: NcUpgraderCtx) {
// fix the existing filter behaviours or
// migrate `null` or `empty` filters to `blank`
await migrateFilters(ncMeta);
// enrich `showNullAndEmptyInFilter` in project meta
// if there is empty / null filters in existing projects,
// then set `showNullAndEmptyInFilter` to true
// else set to false
await updateProjectMeta(ncMeta);
} }

164
packages/nocodb/src/version-upgrader/ncXcdbLTARUpgrader.ts

@ -0,0 +1,164 @@
import { RelationTypes, UITypes } from 'nocodb-sdk';
import NocoCache from '../cache/NocoCache';
import { MetaTable } from '../meta/meta.service';
import { Base } from '../models';
import NcConnectionMgrv2 from '../utils/common/NcConnectionMgrv2';
import { CacheGetType, CacheScope } from '../utils/globals';
import { Model } from '../models';
import type { LinkToAnotherRecordColumn } from '../models';
import type { MetaService } from '../meta/meta.service';
import type { NcUpgraderCtx } from './NcUpgrader';
// An upgrader for upgrading LTAR relations in XCDB bases
// it will delete all the foreign keys and create a new index
// and treat all the LTAR as virtual
async function upgradeModelRelations({
model,
relations,
ncMeta,
sqlClient,
}: {
ncMeta: MetaService;
model: Model;
sqlClient: ReturnType<
(typeof NcConnectionMgrv2)['getSqlClient']
> extends Promise<infer U>
? U
: ReturnType<(typeof NcConnectionMgrv2)['getSqlClient']>;
relations: {
tn: string;
rtn: string;
cn: string;
rcn: string;
}[];
}) {
// Iterate over each column and upgrade LTAR
for (const column of await model.getColumns(ncMeta)) {
if (column.uidt !== UITypes.LinkToAnotherRecord) {
continue;
}
const colOptions = await column.getColOptions<LinkToAnotherRecordColumn>(
ncMeta,
);
switch (colOptions.type) {
case RelationTypes.HAS_MANY:
{
// skip if virtual
if (colOptions.virtual) {
break;
}
const parentCol = await colOptions.getParentColumn(ncMeta);
const childCol = await colOptions.getChildColumn(ncMeta);
const parentModel = await parentCol.getModel(ncMeta);
const childModel = await childCol.getModel(ncMeta);
// delete the foreign key constraint if exists
const relation = relations.find((r) => {
return (
parentCol.column_name === r.rcn &&
childCol.column_name === r.cn &&
parentModel.table_name === r.rtn &&
childModel.table_name === r.tn
);
});
// delete the relation
if (relation) {
await sqlClient.relationDelete({
parentColumn: relation.rcn,
childColumn: relation.cn,
parentTable: relation.rtn,
childTable: relation.tn,
});
}
// skip postgres since we were already creating the index while creating the relation
if (ncMeta.knex.clientType() !== 'pg') {
// create a new index for the column
const indexArgs = {
columns: [relation.cn],
tn: relation.tn,
non_unique: true,
};
await sqlClient.indexCreate(indexArgs);
}
}
break;
}
// update the relation as virtual
await ncMeta.metaUpdate(
null,
null,
MetaTable.COL_RELATIONS,
{ virtual: true },
colOptions.id,
);
// update the cache as well
const cachedData = await NocoCache.get(
`${CacheScope.COL_RELATION}:${colOptions.fk_column_id}`,
CacheGetType.TYPE_OBJECT,
);
if (cachedData) {
cachedData.virtual = true;
await NocoCache.set(
`${CacheScope.COL_RELATION}:${colOptions.fk_column_id}`,
cachedData,
);
}
}
}
// An upgrader for upgrading any existing relation in xcdb
async function upgradeBaseRelations({
ncMeta,
base,
}: {
ncMeta: MetaService;
base: any;
}) {
// const sqlMgr = ProjectMgrv2.getSqlMgr({ id: base.project_id }, ncMeta);
const sqlClient = await NcConnectionMgrv2.getSqlClient(base, ncMeta.knex);
// get all relations
const relations = (await sqlClient.relationListAll())?.data?.list;
// get models for the base
const models = await ncMeta.metaList2(null, base.id, MetaTable.MODELS);
// get all columns and filter out relations and upgrade
for (const model of models) {
await upgradeModelRelations({
ncMeta,
model: new Model(model),
sqlClient,
relations,
});
}
}
// database to virtual relation and create an index for it
export default async function ({ ncMeta }: NcUpgraderCtx) {
// get all xcdb bases
const bases = await ncMeta.metaList2(null, null, MetaTable.BASES, {
condition: {
is_meta: 1,
},
orderBy: {},
});
// iterate and upgrade each base
for (const base of bases) {
await upgradeBaseRelations({
ncMeta,
base: new Base(base),
});
}
}

14
packages/nocodb/tests/unit/rest/tests/tableRow.test.ts

@ -1422,20 +1422,14 @@ function tableTest() {
rowId: row['Id'], rowId: row['Id'],
}); });
const response = await request(context.app) await request(context.app)
.delete(`/api/v1/db/data/noco/${project.id}/${table.id}/${row['Id']}`) .delete(`/api/v1/db/data/noco/${project.id}/${table.id}/${row['Id']}`)
.set('xc-auth', context.token) .set('xc-auth', context.token)
.expect(400); .expect(200);
const deleteRow = await getRow(context, { project, table, id: row['Id'] }); const deleteRow = await getRow(context, { project, table, id: row['Id'] });
if (!deleteRow) { if (deleteRow !== undefined) {
throw new Error('Should not delete'); throw new Error('Record should have been deleted!');
}
if (
!(response.body.msg as string).includes('is a LinkToAnotherRecord of')
) {
throw new Error('Should give LTAR foreign key error');
} }
}); });

24
packages/nocodb/tests/unit/rest/tests/viewRow.test.ts

@ -1,5 +1,4 @@
import 'mocha'; import 'mocha';
import { isString } from 'util';
import request from 'supertest'; import request from 'supertest';
import { UITypes, ViewTypes } from 'nocodb-sdk'; import { UITypes, ViewTypes } from 'nocodb-sdk';
import { expect } from 'chai'; import { expect } from 'chai';
@ -20,7 +19,6 @@ import {
getOneRow, getOneRow,
getRow, getRow,
} from '../../factory/row'; } from '../../factory/row';
import { isPg } from '../../init/db';
import type { ColumnType } from 'nocodb-sdk'; import type { ColumnType } from 'nocodb-sdk';
import type View from '../../../../src/models/View'; import type View from '../../../../src/models/View';
import type Model from '../../../../src/models/Model'; import type Model from '../../../../src/models/Model';
@ -1339,7 +1337,7 @@ function viewRowTests() {
await testDeleteViewRow(ViewTypes.FORM); await testDeleteViewRow(ViewTypes.FORM);
}); });
const testDeleteViewRowWithForiegnKeyConstraint = async ( const testDeleteViewRowWithForeignKeyConstraint = async (
viewType: ViewTypes, viewType: ViewTypes,
) => { ) => {
const table = await createTable(context, project); const table = await createTable(context, project);
@ -1370,35 +1368,29 @@ function viewRowTests() {
rowId: row['Id'], rowId: row['Id'],
}); });
const response = await request(context.app) await request(context.app)
.delete( .delete(
`/api/v1/db/data/noco/${project.id}/${table.id}/views/${view.id}/${row['Id']}`, `/api/v1/db/data/noco/${project.id}/${table.id}/views/${view.id}/${row['Id']}`,
) )
.set('xc-auth', context.token) .set('xc-auth', context.token)
.expect(400); .expect(200);
const deleteRow = await getRow(context, { project, table, id: row['Id'] }); const deleteRow = await getRow(context, { project, table, id: row['Id'] });
if (!deleteRow) { if (deleteRow !== undefined) {
throw new Error('Should not delete'); throw new Error('Record should have been deleted!');
}
if (
!(response.body.msg as string).includes('is a LinkToAnotherRecord of')
) {
throw new Error('Should give ltar foreign key error');
} }
}; };
it('Delete view row with ltar foreign key constraint GALLERY', async function () { it('Delete view row with ltar foreign key constraint GALLERY', async function () {
await testDeleteViewRowWithForiegnKeyConstraint(ViewTypes.GALLERY); await testDeleteViewRowWithForeignKeyConstraint(ViewTypes.GALLERY);
}); });
it('Delete view row with ltar foreign key constraint GRID', async function () { it('Delete view row with ltar foreign key constraint GRID', async function () {
await testDeleteViewRowWithForiegnKeyConstraint(ViewTypes.GRID); await testDeleteViewRowWithForeignKeyConstraint(ViewTypes.GRID);
}); });
it('Delete view row with ltar foreign key constraint FORM', async function () { it('Delete view row with ltar foreign key constraint FORM', async function () {
await testDeleteViewRowWithForiegnKeyConstraint(ViewTypes.FORM); await testDeleteViewRowWithForeignKeyConstraint(ViewTypes.FORM);
}); });
const testViewRowExists = async (viewType: ViewTypes) => { const testViewRowExists = async (viewType: ViewTypes) => {

14
tests/playwright/pages/Dashboard/Grid/index.ts

@ -185,6 +185,12 @@ export class GridPage extends BasePage {
await (await this.rootPage.locator('.ant-drawer-body').elementHandle())?.waitForElementState('stable'); await (await this.rootPage.locator('.ant-drawer-body').elementHandle())?.waitForElementState('stable');
} }
async selectRow(index: number) {
const cell: Locator = await this.get().locator(`td[data-testid="cell-Id-${index}"]`);
await cell.hover();
await cell.locator('input[type="checkbox"]').check({ force: true });
}
async selectAll() { async selectAll() {
await this.get().locator('[data-testid="nc-check-all"]').hover(); await this.get().locator('[data-testid="nc-check-all"]').hover();
@ -201,8 +207,7 @@ export class GridPage extends BasePage {
await this.rootPage.waitForTimeout(300); await this.rootPage.waitForTimeout(300);
} }
async deleteAll() { async deleteSelectedRows() {
await this.selectAll();
await this.get().locator('[data-testid="nc-check-all"]').nth(0).click({ await this.get().locator('[data-testid="nc-check-all"]').nth(0).click({
button: 'right', button: 'right',
}); });
@ -210,6 +215,11 @@ export class GridPage extends BasePage {
await this.dashboard.waitForLoaderToDisappear(); await this.dashboard.waitForLoaderToDisappear();
} }
async deleteAll() {
await this.selectAll();
await this.deleteSelectedRows();
}
async verifyTotalRowCount({ count }: { count: number }) { async verifyTotalRowCount({ count }: { count: number }) {
// wait for 100 ms and try again : 5 times // wait for 100 ms and try again : 5 times
let i = 0; let i = 0;

40
tests/playwright/setup/xcdbProject.ts

@ -0,0 +1,40 @@
import { Api } from 'nocodb-sdk';
let api: Api;
async function createXcdb(token?: string) {
api = new Api({
baseURL: `http://localhost:8080/`,
headers: {
'xc-auth': token,
},
});
const projectList = await api.project.list();
for (const project of projectList.list) {
// delete project with title 'xcdb' if it exists
if (project.title === 'xcdb') {
await api.project.delete(project.id);
}
}
const project = await api.project.create({ title: 'xcdb' });
return project;
}
async function deleteXcdb(token?: string) {
api = new Api({
baseURL: `http://localhost:8080/`,
headers: {
'xc-auth': token,
},
});
const projectList = await api.project.list();
for (const project of projectList.list) {
// delete project with title 'xcdb' if it exists
if (project.title === 'xcdb') {
await api.project.delete(project.id);
}
}
}
export { createXcdb, deleteXcdb };

324
tests/playwright/tests/db/metaLTAR.spec.ts

@ -0,0 +1,324 @@
/*
*
* Meta projects, additional provision for deleting of rows, columns and tables with Link to another record field type
*
* Pre-requisite:
* TableA <hm> TableB <hm> TableC
* TableA <mm> TableD <mm> TableE
* TableA <hm> TableA : self relation
* TableA <mm> TableA : self relation
* Insert some records in TableA, TableB, TableC, TableD, TableE, add some links between them
*
*
* Tests:
* 1. Delete a row from TableA : Verify record status in adjacent tables
* 2. Delete hm link from TableA to TableB : Verify record status in adjacent tables
* 3. Delete mm link from TableA to TableD : Verify record status in adjacent tables
* 4. Delete a self link column from TableA : Verify
* 5. Delete TableA : Verify record status in adjacent tables
*
*/
import { test } from '@playwright/test';
import setup from '../../setup';
import { Api, UITypes } from 'nocodb-sdk';
import { DashboardPage } from '../../pages/Dashboard';
import { GridPage } from '../../pages/Dashboard/Grid';
import { createXcdb, deleteXcdb } from '../../setup/xcdbProject';
import { ProjectsPage } from '../../pages/ProjectsPage';
import { isSqlite } from '../../setup/db';
let api: Api<any>;
const recordCount = 10;
// serial as all projects end up creating xcdb using same name
// fix me : use worker ID logic for creating unique project name
test.describe.serial('Test table', () => {
let context: any;
let dashboard: DashboardPage;
let grid: GridPage;
const tables = [];
test.afterEach(async () => {
try {
if (context) {
await deleteXcdb(context.token);
}
} catch (e) {
console.log(e);
}
// reset tables array
tables.length = 0;
});
test.beforeEach(async ({ page }) => {
context = await setup({ page, isEmptyProject: true });
dashboard = new DashboardPage(page, context.project);
grid = dashboard.grid;
// create a new xcdb project
const xcdb = await createXcdb(context.token);
await dashboard.clickHome();
const projectsPage = new ProjectsPage(dashboard.rootPage);
await projectsPage.openProject({ title: 'xcdb', withoutPrefix: true });
api = new Api({
baseURL: `http://localhost:8080/`,
headers: {
'xc-auth': context.token,
},
});
const columns = [
{
column_name: 'Id',
title: 'Id',
uidt: UITypes.ID,
},
{
column_name: 'Title',
title: 'Title',
uidt: UITypes.SingleLineText,
pv: true,
},
];
const rows = [];
for (let i = 0; i < recordCount * 10; i++) {
rows.push({
Id: i + 1,
Title: `${i + 1}`,
});
}
for (let i = 0; i < 5; i++) {
const table = await api.base.tableCreate(xcdb.id, xcdb.bases?.[0].id, {
table_name: `Table${i}`,
title: `Table${i}`,
columns: columns,
});
tables.push(table);
await api.dbTableRow.bulkCreate('noco', xcdb.id, tables[i].id, rows);
}
// Create links
// TableA <hm> TableB <hm> TableC
await api.dbTableColumn.create(tables[0].id, {
uidt: UITypes.LinkToAnotherRecord,
title: `TableA:hm:TableB`,
column_name: `TableA:hm:TableB`,
parentId: tables[0].id,
childId: tables[1].id,
type: 'hm',
});
await api.dbTableColumn.create(tables[1].id, {
uidt: UITypes.LinkToAnotherRecord,
title: `TableB:hm:TableC`,
column_name: `TableB:hm:TableC`,
parentId: tables[1].id,
childId: tables[2].id,
type: 'hm',
});
// TableA <mm> TableD <mm> TableE
await api.dbTableColumn.create(tables[0].id, {
uidt: UITypes.LinkToAnotherRecord,
title: `TableA:mm:TableD`,
column_name: `TableA:mm:TableD`,
parentId: tables[0].id,
childId: tables[3].id,
type: 'mm',
});
await api.dbTableColumn.create(tables[3].id, {
uidt: UITypes.LinkToAnotherRecord,
title: `TableD:mm:TableE`,
column_name: `TableD:mm:TableE`,
parentId: tables[3].id,
childId: tables[4].id,
type: 'mm',
});
// TableA <hm> TableA : self relation
await api.dbTableColumn.create(tables[0].id, {
uidt: UITypes.LinkToAnotherRecord,
title: `TableA:hm:TableA`,
column_name: `TableA:hm:TableA`,
parentId: tables[0].id,
childId: tables[0].id,
type: 'hm',
});
// TableA <mm> TableA : self relation
await api.dbTableColumn.create(tables[0].id, {
uidt: UITypes.LinkToAnotherRecord,
title: `TableA:mm:TableA`,
column_name: `TableA:mm:TableA`,
parentId: tables[0].id,
childId: tables[0].id,
type: 'mm',
});
// Add links
// TableA <hm> TableB <hm> TableC
// Link every record in tableA to 3 records in tableB
for (let i = 1; i <= recordCount; i++) {
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'hm', 'TableA:hm:TableB', `${i * 3 - 2}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'hm', 'TableA:hm:TableB', `${i * 3 - 1}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'hm', 'TableA:hm:TableB', `${i * 3 - 0}`);
}
// Link every record in tableB to 3 records in tableC
for (let i = 1; i <= recordCount; i++) {
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[1].id, i, 'hm', 'TableB:hm:TableC', `${i * 3 - 2}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[1].id, i, 'hm', 'TableB:hm:TableC', `${i * 3 - 1}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[1].id, i, 'hm', 'TableB:hm:TableC', `${i * 3 - 0}`);
}
// TableA <mm> TableD <mm> TableE
// Link every record in tableA to 5 records in tableD
for (let i = 1; i <= recordCount; i++) {
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableD', `${i}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableD', `${i + 1}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableD', `${i + 2}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableD', `${i + 3}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableD', `${i + 4}`);
}
// Link every record in tableD to 5 records in tableE
for (let i = 1; i <= recordCount; i++) {
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[3].id, i, 'mm', 'TableD:mm:TableE', `${i}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[3].id, i, 'mm', 'TableD:mm:TableE', `${i + 1}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[3].id, i, 'mm', 'TableD:mm:TableE', `${i + 2}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[3].id, i, 'mm', 'TableD:mm:TableE', `${i + 3}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[3].id, i, 'mm', 'TableD:mm:TableE', `${i + 4}`);
}
// TableA <hm> TableA : self relation
// Link every record in tableA to 3 records in tableA
for (let i = 1; i <= recordCount; i++) {
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'hm', 'TableA:hm:TableA', `${i * 3 - 2}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'hm', 'TableA:hm:TableA', `${i * 3 - 1}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'hm', 'TableA:hm:TableA', `${i * 3 - 0}`);
}
// TableA <mm> TableA : self relation
// Link every record in tableA to 5 records in tableA
for (let i = 1; i <= recordCount; i++) {
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableA', `${i}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableA', `${i + 1}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableA', `${i + 2}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableA', `${i + 3}`);
await api.dbTableRow.nestedAdd('noco', xcdb.id, tables[0].id, i, 'mm', 'TableA:mm:TableA', `${i + 4}`);
}
// refresh page
await page.reload();
});
test('Delete record - single, over UI', async () => {
await dashboard.treeView.openTable({ title: 'Table0' });
await grid.deleteRow(0);
// verify row count
await dashboard.grid.verifyTotalRowCount({ count: 99 });
// verify row count in all tables
for (let i = 1; i <= 4; i++) {
await dashboard.treeView.openTable({ title: `Table${i}` });
await dashboard.grid.verifyTotalRowCount({ count: 100 });
}
// has-many removal verification
await dashboard.treeView.openTable({ title: 'Table1' });
await dashboard.grid.cell.verifyVirtualCell({ index: 0, columnHeader: 'Table0', count: 0, value: [] });
await dashboard.grid.cell.verifyVirtualCell({ index: 1, columnHeader: 'Table0', count: 0, value: [] });
await dashboard.grid.cell.verifyVirtualCell({ index: 2, columnHeader: 'Table0', count: 0, value: [] });
// many-many removal verification
await dashboard.treeView.openTable({ title: 'Table3' });
await dashboard.grid.cell.verifyVirtualCell({ index: 0, columnHeader: 'Table0 List', count: 0, value: [] });
await dashboard.grid.cell.verifyVirtualCell({ index: 1, columnHeader: 'Table0 List', count: 1, value: ['2'] });
await dashboard.grid.cell.verifyVirtualCell({ index: 2, columnHeader: 'Table0 List', count: 2, value: ['2', '3'] });
await dashboard.grid.cell.verifyVirtualCell({
index: 3,
columnHeader: 'Table0 List',
count: 3,
value: ['2', '3', '4'],
});
await dashboard.grid.cell.verifyVirtualCell({
index: 4,
columnHeader: 'Table0 List',
count: 4,
value: ['2', '3', '4', '5'],
});
});
test('Delete record - bulk, over UI', async () => {
await dashboard.treeView.openTable({ title: 'Table0' });
await grid.selectRow(0);
await grid.selectRow(1);
await grid.selectRow(2);
await grid.deleteSelectedRows();
// verify row count
await dashboard.grid.verifyTotalRowCount({ count: 97 });
// verify row count in all tables
for (let i = 1; i <= 4; i++) {
await dashboard.treeView.openTable({ title: `Table${i}` });
await dashboard.grid.verifyTotalRowCount({ count: 100 });
}
});
test('Delete column', async () => {
// has-many
await dashboard.treeView.openTable({ title: 'Table0' });
await dashboard.grid.column.delete({ title: 'TableA:hm:TableB' });
// verify
await dashboard.treeView.openTable({ title: 'Table1' });
await dashboard.grid.column.verify({ title: 'Table0', isVisible: false });
await dashboard.grid.column.verify({ title: 'TableB:hm:TableC', isVisible: true });
///////////////////////////////////////////////////////////////////////////////////////////////
// many-many
await dashboard.treeView.openTable({ title: 'Table0' });
await dashboard.grid.column.delete({ title: 'TableA:mm:TableD' });
// verify
await dashboard.treeView.openTable({ title: 'Table3' });
await dashboard.grid.column.verify({ title: 'Table0 List', isVisible: false });
await dashboard.grid.column.verify({ title: 'TableD:mm:TableE', isVisible: true });
///////////////////////////////////////////////////////////////////////////////////////////////
// has-many self relation
await dashboard.treeView.openTable({ title: 'Table0' });
await dashboard.grid.column.delete({ title: 'TableA:hm:TableA' });
// verify
await dashboard.grid.column.verify({ title: 'TableA:hm:TableA', isVisible: false });
await dashboard.grid.column.verify({ title: 'Table0', isVisible: false });
///////////////////////////////////////////////////////////////////////////////////////////////
// many-many self relation
await dashboard.treeView.openTable({ title: 'Table0' });
await dashboard.grid.column.delete({ title: 'TableA:mm:TableA' });
// verify
await dashboard.grid.column.verify({ title: 'Table0 List', isVisible: false });
await dashboard.grid.column.verify({ title: 'TableA:mm:TableA', isVisible: false });
});
test('Delete table', async () => {
await dashboard.treeView.deleteTable({ title: 'Table0' });
await dashboard.treeView.verifyTable({ title: 'Table0', exists: false });
// verify
await dashboard.treeView.openTable({ title: 'Table1' });
await dashboard.grid.column.verify({ title: 'Table0', isVisible: false });
await dashboard.treeView.openTable({ title: 'Table3' });
await dashboard.grid.column.verify({ title: 'Table0 List', isVisible: false });
});
});

25
tests/playwright/tests/db/timezone.spec.ts

@ -30,7 +30,7 @@ const rowAttributes = [
{ Id: 3, DateTime: '2020-12-31 20:00:00-04:00' }, { Id: 3, DateTime: '2020-12-31 20:00:00-04:00' },
]; ];
async function timezoneSuite(token?: string, skipTableCreate?: boolean) { async function timezoneSuite(token: string, projectTitle: string, skipTableCreate?: boolean) {
api = new Api({ api = new Api({
baseURL: `http://localhost:8080/`, baseURL: `http://localhost:8080/`,
headers: { headers: {
@ -41,12 +41,12 @@ async function timezoneSuite(token?: string, skipTableCreate?: boolean) {
const projectList = await api.project.list(); const projectList = await api.project.list();
for (const project of projectList.list) { for (const project of projectList.list) {
// delete project with title 'xcdb' if it exists // delete project with title 'xcdb' if it exists
if (project.title === 'xcdb') { if (project.title === projectTitle) {
await api.project.delete(project.id); await api.project.delete(project.id);
} }
} }
const project = await api.project.create({ title: 'xcdb' }); const project = await api.project.create({ title: projectTitle });
if (skipTableCreate) return { project }; if (skipTableCreate) return { project };
const table = await api.base.tableCreate(project.id, project.bases?.[0].id, { const table = await api.base.tableCreate(project.id, project.bases?.[0].id, {
table_name: 'dateTimeTable', table_name: 'dateTimeTable',
@ -99,7 +99,8 @@ async function connectToExtDb(context: any, dbName: string) {
//////////////////////////////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////////////
test.describe('Timezone-XCDB : Japan/Tokyo', () => { // serial : as we are creating an external db, we need to run the tests sequentially
test.describe.serial('Timezone-XCDB : Japan/Tokyo', () => {
let dashboard: DashboardPage; let dashboard: DashboardPage;
let context: any; let context: any;
@ -109,7 +110,7 @@ test.describe('Timezone-XCDB : Japan/Tokyo', () => {
if (!isSqlite(context)) return; if (!isSqlite(context)) return;
try { try {
const { project, table } = await timezoneSuite(context.token); const { project, table } = await timezoneSuite(context.token, 'xcdb0');
await api.dbTableRow.bulkCreate('noco', project.id, table.id, rowAttributes); await api.dbTableRow.bulkCreate('noco', project.id, table.id, rowAttributes);
records = await api.dbTableRow.list('noco', project.id, table.id, { limit: 10 }); records = await api.dbTableRow.list('noco', project.id, table.id, { limit: 10 });
@ -143,7 +144,7 @@ test.describe('Timezone-XCDB : Japan/Tokyo', () => {
await dashboard.clickHome(); await dashboard.clickHome();
const projectsPage = new ProjectsPage(dashboard.rootPage); const projectsPage = new ProjectsPage(dashboard.rootPage);
await projectsPage.openProject({ title: 'xcdb', withoutPrefix: true }); await projectsPage.openProject({ title: 'xcdb0', withoutPrefix: true });
await dashboard.treeView.openTable({ title: 'dateTimeTable' }); await dashboard.treeView.openTable({ title: 'dateTimeTable' });
@ -202,7 +203,7 @@ test.describe('Timezone-XCDB : Japan/Tokyo', () => {
// Change browser timezone & locale to Asia/Hong-Kong // Change browser timezone & locale to Asia/Hong-Kong
// //
test.describe('Timezone-XCDB : Asia/Hong-kong', () => { test.describe.serial('Timezone-XCDB : Asia/Hong-kong', () => {
let dashboard: DashboardPage; let dashboard: DashboardPage;
let context: any; let context: any;
@ -211,7 +212,7 @@ test.describe('Timezone-XCDB : Asia/Hong-kong', () => {
dashboard = new DashboardPage(page, context.project); dashboard = new DashboardPage(page, context.project);
try { try {
const { project, table } = await timezoneSuite(context.token); const { project, table } = await timezoneSuite(context.token, 'xcdb1');
await api.dbTableRow.bulkCreate('noco', project.id, table.id, rowAttributes); await api.dbTableRow.bulkCreate('noco', project.id, table.id, rowAttributes);
} catch (e) { } catch (e) {
console.error(e); console.error(e);
@ -240,7 +241,7 @@ test.describe('Timezone-XCDB : Asia/Hong-kong', () => {
test('API insert, verify display value', async () => { test('API insert, verify display value', async () => {
await dashboard.clickHome(); await dashboard.clickHome();
const projectsPage = new ProjectsPage(dashboard.rootPage); const projectsPage = new ProjectsPage(dashboard.rootPage);
await projectsPage.openProject({ title: 'xcdb', withoutPrefix: true }); await projectsPage.openProject({ title: 'xcdb1', withoutPrefix: true });
await dashboard.treeView.openTable({ title: 'dateTimeTable' }); await dashboard.treeView.openTable({ title: 'dateTimeTable' });
@ -268,7 +269,7 @@ test.describe('Timezone-XCDB : Asia/Hong-kong', () => {
//////////////////////////////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////////////
test.describe('Timezone-XCDB : Asia/Hong-kong', () => { test.describe.serial('Timezone-XCDB : Asia/Hong-kong', () => {
let dashboard: DashboardPage; let dashboard: DashboardPage;
let context: any; let context: any;
@ -284,7 +285,7 @@ test.describe('Timezone-XCDB : Asia/Hong-kong', () => {
// Apply only for sqlite, as currently- root DB for all instances is SQLite // Apply only for sqlite, as currently- root DB for all instances is SQLite
if (!isSqlite(context)) return; if (!isSqlite(context)) return;
const { project } = await timezoneSuite(context.token, true); const { project } = await timezoneSuite(context.token, 'xcdb2', true);
context.project = project; context.project = project;
// Kludge: Using API for test preparation was not working // Kludge: Using API for test preparation was not working
@ -292,7 +293,7 @@ test.describe('Timezone-XCDB : Asia/Hong-kong', () => {
await dashboard.clickHome(); await dashboard.clickHome();
const projectsPage = new ProjectsPage(dashboard.rootPage); const projectsPage = new ProjectsPage(dashboard.rootPage);
await projectsPage.openProject({ title: 'xcdb', withoutPrefix: true }); await projectsPage.openProject({ title: 'xcdb2', withoutPrefix: true });
await dashboard.treeView.createTable({ title: 'dateTimeTable', mode: 'Xcdb' }); await dashboard.treeView.createTable({ title: 'dateTimeTable', mode: 'Xcdb' });
await dashboard.grid.column.create({ await dashboard.grid.column.create({

Loading…
Cancel
Save