mirror of https://github.com/nocodb/nocodb
Pranav C
3 years ago
20 changed files with 661 additions and 80 deletions
@ -0,0 +1,38 @@
|
||||
<template> |
||||
<div> |
||||
<v-icon small @click="metas = true"> |
||||
mdi-bug-outline |
||||
</v-icon> |
||||
|
||||
<v-dialog v-model="metas" max-width="800px"> |
||||
<v-card> |
||||
<v-tabs height="30"> |
||||
<v-tab v-for="(meta,table) in metasObj" :key="table"> |
||||
<span class="caption text-capitalize">{{ table }}</span> |
||||
</v-tab> |
||||
<v-tab-item v-for="(meta,table) in metasObj" :key="table"> |
||||
<monaco-json-object-editor :value="meta" style="height:80vh" /> |
||||
</v-tab-item> |
||||
</v-tabs> |
||||
</v-card> |
||||
</v-dialog> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import MonacoJsonObjectEditor from '@/components/monaco/MonacoJsonObjectEditor' |
||||
export default { |
||||
name: 'DebugMetas', |
||||
components: { MonacoJsonObjectEditor }, |
||||
data: () => ({ metas: false, tab: 0 }), |
||||
computed: { |
||||
metasObj() { |
||||
return this.$store.state.meta.metas || {} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,34 @@
|
||||
import RestApi from './restApi' |
||||
import GqlApi from './gqlApi' |
||||
|
||||
export default class ApiFactory { |
||||
static create(table, type, ctx) { |
||||
if (type === 'graphql') { |
||||
return new GqlApi(table, ctx) |
||||
} else if (type === 'rest') { |
||||
return new RestApi(table, ctx) |
||||
} |
||||
} |
||||
} |
||||
/** |
||||
* @copyright Copyright (c) 2021, Xgene Cloud Ltd |
||||
* |
||||
* @author Naveen MR <oof1lab@gmail.com> |
||||
* @author Pranav C Balan <pranavxc@gmail.com> |
||||
* |
||||
* @license GNU AGPL version 3 or any later version |
||||
* |
||||
* This program is free software: you can redistribute it and/or modify |
||||
* it under the terms of the GNU Affero General Public License as |
||||
* published by the Free Software Foundation, either version 3 of the |
||||
* License, or (at your option) any later version. |
||||
* |
||||
* This program is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
* GNU Affero General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Affero General Public License |
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
* |
||||
*/ |
@ -0,0 +1,263 @@
|
||||
export default class GqlApi { |
||||
constructor(table, $ctx) { |
||||
this.$ctx = $ctx |
||||
this.table = $ctx.$store.state.meta.metas[table]._tn |
||||
} |
||||
|
||||
get meta() { |
||||
return this.$ctx.$store.state.meta.metas[this.$ctx.table] || {} |
||||
} |
||||
|
||||
get columns() { |
||||
return this.meta.columns || [] |
||||
} |
||||
|
||||
// todo: - get version letter and use table alias
|
||||
async list(params) { |
||||
const data = await this.post(`/nc/${this.$ctx.projectId}/v1/graphql`, { |
||||
query: await this.gqlQuery(params), |
||||
variables: null |
||||
}) |
||||
return data.data.data[this.gqlQueryListName] |
||||
} |
||||
|
||||
async count(params) { |
||||
const data = await this.post(`/nc/${this.$ctx.projectId}/v1/graphql`, { |
||||
query: this.gqlCountQuery(params), |
||||
variables: null |
||||
}) |
||||
return data.data.data[this.gqlQueryCountName] |
||||
} |
||||
|
||||
post(url, params) { |
||||
return this.$axios({ |
||||
url: `${this.$axios.defaults.baseURL}${url}`, |
||||
method: 'post', |
||||
data: params |
||||
}) |
||||
} |
||||
|
||||
generateQueryParams(params) { |
||||
if (!params) { return '(where:"")' } |
||||
const res = [] |
||||
if ('limit' in params) { |
||||
res.push(`limit: ${params.limit}`) |
||||
} |
||||
if ('offset' in params) { |
||||
res.push(`offset: ${params.offset}`) |
||||
} |
||||
if ('where' in params) { |
||||
res.push(`where: ${JSON.stringify(params.where)}`) |
||||
} |
||||
if ('sort' in params) { |
||||
res.push(`sort: ${JSON.stringify(params.sort)}`) |
||||
} |
||||
if (params.condition) { |
||||
res.push(`condition: ${JSON.stringify(params.condition)}`) |
||||
} |
||||
if (params.conditionGraph) { |
||||
res.push(`conditionGraph: ${JSON.stringify(JSON.stringify(params.conditionGraph))}`) |
||||
} |
||||
return `(${res.join(',')})` |
||||
} |
||||
|
||||
async gqlQuery(params) { |
||||
return `{${this.gqlQueryListName}${this.generateQueryParams(params)}{${this.gqlReqBody}${await this.gqlRelationReqBody(params)}}}` |
||||
} |
||||
|
||||
gqlReadQuery(id) { |
||||
return `{${this.gqlQueryReadName}(id:"${id}"){${this.gqlReqBody}}}` |
||||
} |
||||
|
||||
gqlCountQuery(params) { |
||||
return `{${this.gqlQueryCountName}${this.generateQueryParams(params)}}` |
||||
} |
||||
|
||||
get gqlQueryListName() { |
||||
return `${this.meta._tn}List` |
||||
} |
||||
|
||||
get gqlQueryReadName() { |
||||
return `${this.meta._tn}Read` |
||||
} |
||||
|
||||
get tableCamelized() { |
||||
return `${this.meta._tn}` |
||||
} |
||||
|
||||
get gqlReqBody() { |
||||
return `\n${this.columns.map(c => c._cn).join('\n')}\n` |
||||
} |
||||
|
||||
async gqlRelationReqBody(params) { |
||||
let str = '' |
||||
if (params.hm) { |
||||
for (const child of params.hm.split(',')) { |
||||
await this.$ctx.$store.dispatch('meta/ActLoadMeta', { |
||||
dbAlias: this.$ctx.dbAlias, |
||||
env: this.$ctx.env, |
||||
tn: child |
||||
}) |
||||
const meta = this.$ctx.$store.state.meta.metas[child] |
||||
if (meta) { |
||||
str += `\n${meta._tn}List{\n${meta.columns.map(c => c._cn).join('\n')}\n}` |
||||
} |
||||
} |
||||
} |
||||
if (params.bt) { |
||||
for (const parent of params.bt.split(',')) { |
||||
await this.$ctx.$store.dispatch('meta/ActLoadMeta', { |
||||
dbAlias: this.$ctx.dbAlias, |
||||
env: this.$ctx.env, |
||||
tn: parent |
||||
}) |
||||
const meta = this.$ctx.$store.state.meta.metas[parent] |
||||
if (meta) { |
||||
str += `\n${meta._tn}Read{\n${meta.columns.map(c => c._cn).join('\n')}\n}` |
||||
} |
||||
} |
||||
} |
||||
if (params.mm) { |
||||
for (const mm of params.mm.split(',')) { |
||||
await this.$ctx.$store.dispatch('meta/ActLoadMeta', { |
||||
dbAlias: this.$ctx.dbAlias, |
||||
env: this.$ctx.env, |
||||
tn: mm |
||||
}) |
||||
const meta = this.$ctx.$store.state.meta.metas[mm] |
||||
if (meta) { |
||||
str += `\n${meta._tn}MMList{\n${meta.columns.map(c => c._cn).join('\n')}\n}` |
||||
} |
||||
} |
||||
} |
||||
return str |
||||
} |
||||
|
||||
get gqlQueryCountName() { |
||||
return `${this.tableCamelized}Count` |
||||
} |
||||
|
||||
get gqlMutationCreateName() { |
||||
return `${this.tableCamelized}Create` |
||||
} |
||||
|
||||
get gqlMutationUpdateName() { |
||||
return `${this.tableCamelized}Update` |
||||
} |
||||
|
||||
get gqlMutationDeleteName() { |
||||
return `${this.tableCamelized}Delete` |
||||
} |
||||
|
||||
async paginatedList(params) { |
||||
// const list = await this.list(params);
|
||||
// const count = (await this.count({where: params.where || ''}));
|
||||
const [list, count] = await Promise.all([ |
||||
this.list(params), this.count({ |
||||
where: params.where || '', |
||||
conditionGraph: params.conditionGraph, |
||||
condition: params.condition |
||||
}) |
||||
]) |
||||
return { list, count } |
||||
} |
||||
|
||||
async update(id, data, oldData) { |
||||
const data1 = await this.post(`/nc/${this.$ctx.projectId}/v1/graphql`, { |
||||
query: `mutation update($id:String!, $data:${this.tableCamelized}Input){
|
||||
${this.gqlMutationUpdateName}(id: $id, data: $data) |
||||
}`,
|
||||
variables: { |
||||
id, data |
||||
} |
||||
}) |
||||
|
||||
const colName = Object.keys(data)[0] |
||||
this.$ctx.$store.dispatch('sqlMgr/ActSqlOp', [{ dbAlias: this.$ctx.dbAlias }, 'xcAuditCreate', { |
||||
tn: this.table, |
||||
cn: colName, |
||||
pk: id, |
||||
value: data[colName], |
||||
prevValue: oldData[colName] |
||||
}]) |
||||
|
||||
return data1.data.data[this.gqlMutationUpdateName] |
||||
} |
||||
|
||||
async insert(data) { |
||||
const data1 = await this.post(`/nc/${this.$ctx.projectId}/v1/graphql`, { |
||||
query: `mutation create($data:${this.tableCamelized}Input){
|
||||
${this.gqlMutationCreateName}(data: $data){${this.gqlReqBody}} |
||||
}`,
|
||||
variables: { |
||||
data |
||||
} |
||||
}) |
||||
return data1.data.data[this.gqlMutationCreateName] |
||||
} |
||||
|
||||
async delete(id) { |
||||
const data1 = await this.post(`/nc/${this.$ctx.projectId}/v1/graphql`, { |
||||
query: `mutation delete($id:String!){
|
||||
${this.gqlMutationDeleteName}(id: $id) |
||||
}`,
|
||||
variables: { id } |
||||
}) |
||||
|
||||
return data1.data.data[this.gqlMutationDeleteName] |
||||
} |
||||
|
||||
async read(id) { |
||||
const data = await this.post(`/nc/${this.$ctx.projectId}/v1/graphql`, { |
||||
query: this.gqlReadQuery(id), |
||||
variables: null |
||||
}) |
||||
return data.data.data[this.gqlQueryReadName] |
||||
} |
||||
|
||||
get $axios() { |
||||
return this.$ctx.$axios |
||||
} |
||||
|
||||
async paginatedM2mNotChildrenList(params, assoc, pid) { |
||||
const list = await this.post(`/nc/${this.$ctx.projectId}/v1/graphql`, { |
||||
query: `query m2mNotChildren($pid: String!,$assoc:String!,$parent:String!, $limit:Int, $offset:Int){
|
||||
m2mNotChildren(pid: $pid,assoc:$assoc,parent:$parent,limit:$limit, offset:$offset) |
||||
}`,
|
||||
variables: { |
||||
parent: this.meta.tn, assoc, pid: pid + '', ...params |
||||
} |
||||
}) |
||||
const count = await this.post(`/nc/${this.$ctx.projectId}/v1/graphql`, { |
||||
query: `query m2mNotChildrenCount($pid: String!,$assoc:String!,$parent:String!){
|
||||
m2mNotChildrenCount(pid: $pid,assoc:$assoc,parent:$parent) |
||||
}`,
|
||||
variables: { |
||||
parent: this.meta.tn, assoc, pid: pid + '' |
||||
} |
||||
}) |
||||
return { list: list.data.data.m2mNotChildren, count: count.data.data.m2mNotChildrenCount.count } |
||||
} |
||||
} |
||||
/** |
||||
* @copyright Copyright (c) 2021, Xgene Cloud Ltd |
||||
* |
||||
* @author Naveen MR <oof1lab@gmail.com> |
||||
* @author Pranav C Balan <pranavxc@gmail.com> |
||||
* |
||||
* @license GNU AGPL version 3 or any later version |
||||
* |
||||
* This program is free software: you can redistribute it and/or modify |
||||
* it under the terms of the GNU Affero General Public License as |
||||
* published by the Free Software Foundation, either version 3 of the |
||||
* License, or (at your option) any later version. |
||||
* |
||||
* This program is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
* GNU Affero General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Affero General Public License |
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
* |
||||
*/ |
@ -0,0 +1,35 @@
|
||||
import ApiFactory from '@/plugins/ncApis/apiFactory' |
||||
|
||||
export default function({ store: $store, $axios, ...rest }, inject) { |
||||
let instanceRefs = {} |
||||
let projectId = null |
||||
|
||||
inject('ncApis', { |
||||
get: ({ table, dbAlias, env }) => { |
||||
if (!$store.state.meta.metas[table]) { |
||||
return |
||||
} |
||||
|
||||
if (instanceRefs[env] && instanceRefs[env][dbAlias] && instanceRefs[env][dbAlias][table]) { |
||||
return instanceRefs[env][dbAlias][table] |
||||
} |
||||
|
||||
instanceRefs[env] = instanceRefs[env] || {} |
||||
instanceRefs[env][dbAlias] = instanceRefs[env][dbAlias] || {} |
||||
|
||||
instanceRefs[env][dbAlias][table] = ApiFactory.create( |
||||
table, |
||||
$store.getters['project/GtrProjectType'], |
||||
{ $store, $axios, projectId, dbAlias, env, table } |
||||
) |
||||
|
||||
return instanceRefs[env][dbAlias][table] |
||||
}, |
||||
clear: () => { |
||||
instanceRefs = {} |
||||
}, |
||||
setProjectId: (_projectId) => { |
||||
projectId = _projectId |
||||
} |
||||
}) |
||||
} |
@ -0,0 +1,128 @@
|
||||
export default class RestApi { |
||||
constructor(table, $ctx) { |
||||
this.$ctx = $ctx |
||||
this.table = $ctx.$store.state.meta.metas[table]._tn |
||||
} |
||||
|
||||
// todo: - get version letter and use table alias
|
||||
async list(params) { |
||||
// const data = await this.get(`/nc/${this.$ctx.projectId}/api/v1/${this.table}`, params)
|
||||
const data = await this.get(`/nc/${this.$ctx.projectId}/api/v1/${this.table}`, params) |
||||
return data.data |
||||
} |
||||
|
||||
async read(id) { |
||||
const data = await this.get(`/nc/${this.$ctx.projectId}/api/v1/${this.table}/${id}`) |
||||
return data.data |
||||
} |
||||
|
||||
async count(params) { |
||||
if (this.timeout) { |
||||
return this.timeout |
||||
} |
||||
try { |
||||
const data = await this.get(`/nc/${this.$ctx.projectId}/api/v1/${this.table}/count`, params, { |
||||
timeout: 10000 |
||||
}) |
||||
return data && data.data |
||||
} catch (e) { |
||||
if (e.code === 'ECONNABORTED') { |
||||
// eslint-disable-next-line no-return-assign
|
||||
return this.timeout = { count: Infinity } |
||||
} else { |
||||
throw e |
||||
} |
||||
} |
||||
} |
||||
|
||||
get(url, params, extras = {}) { |
||||
return this.$axios({ |
||||
url, |
||||
params, |
||||
...extras |
||||
}) |
||||
} |
||||
|
||||
async paginatedList(params) { |
||||
// const list = await this.list(params);
|
||||
// const count = (await this.count({where: params.where || ''})).count;
|
||||
const [list, { count }] = await Promise.all([this.list(params), this.count({ |
||||
where: params.where || '', |
||||
conditionGraph: params.conditionGraph |
||||
})]) |
||||
return { list, count } |
||||
} |
||||
|
||||
async paginatedM2mNotChildrenList(params, assoc, pid) { |
||||
/// api/v1/Film/m2mNotChildren/film_actor/44
|
||||
// const list = await this.list(params);
|
||||
// const count = (await this.count({where: params.where || ''})).count;
|
||||
const { list, info: { count } } = (await this.get(`/nc/${this.$ctx.projectId}/api/v1/${this.table}/m2mNotChildren/${assoc}/${pid}`, params)).data |
||||
return { list, count } |
||||
} |
||||
|
||||
async update(id, data, oldData) { |
||||
const res = await this.$axios({ |
||||
method: 'put', |
||||
url: `/nc/${this.$ctx.projectId}/api/v1/${this.table}/${id}`, |
||||
data |
||||
}) |
||||
const colName = Object.keys(data)[0] |
||||
this.$ctx.$store.dispatch('sqlMgr/ActSqlOp', [{ dbAlias: this.$ctx.dbAlias }, 'xcAuditCreate', { |
||||
tn: this.table, |
||||
cn: colName, |
||||
pk: id, |
||||
value: data[colName], |
||||
prevValue: oldData[colName] |
||||
}]) |
||||
|
||||
return res |
||||
} |
||||
|
||||
async insert(data) { |
||||
return (await this.$axios({ |
||||
method: 'post', |
||||
url: `/nc/${this.$ctx.projectId}/api/v1/${this.table}`, |
||||
data |
||||
})).data |
||||
} |
||||
|
||||
async delete(id) { |
||||
return this.$axios({ |
||||
method: 'delete', |
||||
url: `/nc/${this.$ctx.projectId}/api/v1/${this.table}/${id}` |
||||
}) |
||||
} |
||||
|
||||
get $axios() { |
||||
return this.$ctx.$axios |
||||
} |
||||
|
||||
get apiUrl() { |
||||
return `${process.env.NODE_ENV === 'production' |
||||
? `${window.location.protocol}//${window.location.hostname}${window.location.port ? `:${window.location.port}` : ''}` |
||||
: 'http://localhost:8080'}/nc/${this.$ctx.projectId}/api/v1/${this.table}` |
||||
} |
||||
} |
||||
/** |
||||
* @copyright Copyright (c) 2021, Xgene Cloud Ltd |
||||
* |
||||
* @author Naveen MR <oof1lab@gmail.com> |
||||
* @author Pranav C Balan <pranavxc@gmail.com> |
||||
* |
||||
* @license GNU AGPL version 3 or any later version |
||||
* |
||||
* This program is free software: you can redistribute it and/or modify |
||||
* it under the terms of the GNU Affero General Public License as |
||||
* published by the Free Software Foundation, either version 3 of the |
||||
* License, or (at your option) any later version. |
||||
* |
||||
* This program is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
* GNU Affero General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Affero General Public License |
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
* |
||||
*/ |
Loading…
Reference in new issue