Browse Source

fix: move sanitize logic to exec (#8585)

* fix: sanitise sql query in bulk insert

* fix: avoid sanitizing column id

* fix: move sanitizeQuery to exec

* fix: improve condition

---------

Co-authored-by: Pranav C <pranavxc@gmail.com>
fix/comments-added-through-api-token-request-cannot-be-displayed
Mert E 6 months ago committed by GitHub
parent
commit
59967dbadc
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
  1. 95
      packages/nocodb/src/db/BaseModelSqlv2.ts

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

@ -40,6 +40,7 @@ import type {
SelectOption, SelectOption,
User, User,
} from '~/models'; } from '~/models';
import type CustomKnex from '~/db/CustomKnex';
import { import {
Audit, Audit,
BaseUser, BaseUser,
@ -542,12 +543,7 @@ class BaseModelSqlv2 {
as: 'count', as: 'count',
}).first(); }).first();
let sql = sanitize(qb.toQuery()); return (await this.execAndParse(qb, null, { raw: true, first: true }))
if (!this.isPg && !this.isMssql && !this.isSnowflake) {
sql = unsanitize(qb.toQuery());
}
return (await this.execAndParse(sql, null, { raw: true, first: true }))
?.count; ?.count;
} }
@ -670,9 +666,9 @@ class BaseModelSqlv2 {
knex: this.dbDriver, knex: this.dbDriver,
columnOptions: (await column.getColOptions()) as RollupColumn, columnOptions: (await column.getColOptions()) as RollupColumn,
}) })
).builder.as(sanitize(column.id)), ).builder.as(column.id),
); );
groupBySelectors.push(sanitize(column.id)); groupBySelectors.push(column.id);
break; break;
case UITypes.Formula: case UITypes.Formula:
{ {
@ -684,14 +680,12 @@ class BaseModelSqlv2 {
selectQb = this.dbDriver.raw(`?? as ??`, [ selectQb = this.dbDriver.raw(`?? as ??`, [
_selectQb.builder, _selectQb.builder,
sanitize(column.id), column.id,
]); ]);
} catch (e) { } catch (e) {
logger.log(e); logger.log(e);
// return dummy select // return dummy select
selectQb = this.dbDriver.raw(`'ERR' as ??`, [ selectQb = this.dbDriver.raw(`'ERR' as ??`, [column.id]);
sanitize(column.id),
]);
} }
selectors.push(selectQb); selectors.push(selectQb);
@ -711,11 +705,11 @@ class BaseModelSqlv2 {
const selectQb = this.dbDriver.raw(`?? as ??`, [ const selectQb = this.dbDriver.raw(`?? as ??`, [
this.dbDriver.raw(_selectQb.builder).wrap('(', ')'), this.dbDriver.raw(_selectQb.builder).wrap('(', ')'),
sanitize(column.id), column.id,
]); ]);
selectors.push(selectQb); selectors.push(selectQb);
groupBySelectors.push(sanitize(column.id)); groupBySelectors.push(column.id);
} }
break; break;
default: default:
@ -724,7 +718,7 @@ class BaseModelSqlv2 {
selectors.push( selectors.push(
this.dbDriver.raw('?? as ??', [columnName, column.id]), this.dbDriver.raw('?? as ??', [columnName, column.id]),
); );
groupBySelectors.push(sanitize(column.id)); groupBySelectors.push(column.id);
} }
break; break;
} }
@ -904,9 +898,9 @@ class BaseModelSqlv2 {
// alias, // alias,
columnOptions: (await column.getColOptions()) as RollupColumn, columnOptions: (await column.getColOptions()) as RollupColumn,
}) })
).builder.as(sanitize(column.id)), ).builder.as(column.id),
); );
groupBySelectors.push(sanitize(column.id)); groupBySelectors.push(column.id);
break; break;
case UITypes.Formula: { case UITypes.Formula: {
let selectQb; let selectQb;
@ -917,14 +911,12 @@ class BaseModelSqlv2 {
selectQb = this.dbDriver.raw(`?? as ??`, [ selectQb = this.dbDriver.raw(`?? as ??`, [
_selectQb.builder, _selectQb.builder,
sanitize(column.id), column.id,
]); ]);
} catch (e) { } catch (e) {
logger.log(e); logger.log(e);
// return dummy select // return dummy select
selectQb = this.dbDriver.raw(`'ERR' as ??`, [ selectQb = this.dbDriver.raw(`'ERR' as ??`, [column.id]);
sanitize(column.id),
]);
} }
selectors.push(selectQb); selectors.push(selectQb);
@ -944,11 +936,11 @@ class BaseModelSqlv2 {
const selectQb = this.dbDriver.raw(`?? as ??`, [ const selectQb = this.dbDriver.raw(`?? as ??`, [
this.dbDriver.raw(_selectQb.builder).wrap('(', ')'), this.dbDriver.raw(_selectQb.builder).wrap('(', ')'),
sanitize(column.id), column.id,
]); ]);
selectors.push(selectQb); selectors.push(selectQb);
groupBySelectors.push(sanitize(column.id)); groupBySelectors.push(column.id);
} }
break; break;
default: default:
@ -957,7 +949,7 @@ class BaseModelSqlv2 {
selectors.push( selectors.push(
this.dbDriver.raw('?? as ??', [columnName, column.id]), this.dbDriver.raw('?? as ??', [columnName, column.id]),
); );
groupBySelectors.push(sanitize(column.id)); groupBySelectors.push(column.id);
} }
break; break;
} }
@ -2638,17 +2630,12 @@ class BaseModelSqlv2 {
aliasToColumnBuilder, aliasToColumnBuilder,
); );
qb.select( qb.select(
this.dbDriver.raw(`?? as ??`, [ this.dbDriver.raw(`?? as ??`, [selectQb.builder, column.id]),
selectQb.builder,
sanitize(column.id),
]),
); );
} catch (e) { } catch (e) {
logger.log(e); logger.log(e);
// return dummy select // return dummy select
qb.select( qb.select(this.dbDriver.raw(`'ERR' as ??`, [column.id]));
this.dbDriver.raw(`'ERR' as ??`, [sanitize(column.id)]),
);
} }
} }
break; break;
@ -2664,7 +2651,7 @@ class BaseModelSqlv2 {
alias, alias,
columnOptions: (await column.getColOptions()) as RollupColumn, columnOptions: (await column.getColOptions()) as RollupColumn,
}) })
).builder.as(sanitize(column.id)), ).builder.as(column.id),
); );
break; break;
case UITypes.CreatedBy: case UITypes.CreatedBy:
@ -5203,6 +5190,26 @@ class BaseModelSqlv2 {
return await this.execAndParse(qb); return await this.execAndParse(qb);
} }
public async execAndGetRows(query: string, trx?: Knex | CustomKnex) {
trx = trx || this.dbDriver;
query = this.sanitizeQuery(query);
if (this.isPg || this.isSnowflake) {
return (await trx.raw(query))?.rows;
} else if (!this.isMssql && /^(\(|)select/i.test(query)) {
return await trx.from(trx.raw(query).wrap('(', ') __nc_alias'));
} else if (this.isMySQL && /^(\(|)insert/i.test(query)) {
const res = await trx.raw(query);
if (res && res[0] && res[0].insertId) {
return res[0].insertId;
}
return res;
} else {
return await trx.raw(query);
}
}
public async execAndParse( public async execAndParse(
qb: Knex.QueryBuilder | string, qb: Knex.QueryBuilder | string,
dependencyColumns?: Column[], dependencyColumns?: Column[],
@ -5233,21 +5240,9 @@ class BaseModelSqlv2 {
qb = qb.limit(1); qb = qb.limit(1);
} }
let query = typeof qb === 'string' ? qb : qb.toQuery(); const query = typeof qb === 'string' ? qb : qb.toQuery();
if (!this.isPg && !this.isMssql && !this.isSnowflake) {
query = unsanitize(query);
} else {
query = sanitize(query);
}
let data = let data = await this.execAndGetRows(query);
this.isPg || this.isSnowflake
? (await this.dbDriver.raw(query))?.rows
: /^(\(|)select/.test(query) && !this.isMssql
? await this.dbDriver.from(
this.dbDriver.raw(query).wrap('(', ') __nc_alias'),
)
: await this.dbDriver.raw(query);
if (!this.model?.columns) { if (!this.model?.columns) {
await this.model.getColumns(); await this.model.getColumns();
@ -5282,6 +5277,14 @@ class BaseModelSqlv2 {
return data; return data;
} }
protected sanitizeQuery(query: string) {
if (!this.isPg && !this.isMssql && !this.isSnowflake) {
return unsanitize(query);
} else {
return sanitize(query);
}
}
async runOps(ops: Promise<string>[], trx = this.dbDriver) { async runOps(ops: Promise<string>[], trx = this.dbDriver) {
const queries = await Promise.all(ops); const queries = await Promise.all(ops);
for (const query of queries) { for (const query of queries) {

Loading…
Cancel
Save