Merge branch 'master' of github.com:Budibase/budibase into cheeks-bugfixes
This commit is contained in:
commit
5a086d529b
|
@ -145,7 +145,7 @@ The HTML and CSS for your apps runtime pages, as well as the budibase client lib
|
||||||
|
|
||||||
#### Backend
|
#### Backend
|
||||||
|
|
||||||
The backend schema, models and records are stored using PouchDB when developing locally, and in [CouchDB](https://pouchdb.com/) when running in production.
|
The backend schema, tables and rows are stored using PouchDB when developing locally, and in [CouchDB](https://pouchdb.com/) when running in production.
|
||||||
|
|
||||||
### Publishing Budibase to NPM
|
### Publishing Budibase to NPM
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
{
|
{
|
||||||
"version": "0.2.0",
|
"version": "0.2.1",
|
||||||
"npmClient": "yarn",
|
"npmClient": "yarn",
|
||||||
"packages": [
|
"packages": [
|
||||||
"packages/*"
|
"packages/*"
|
||||||
|
|
|
@ -226,7 +226,7 @@
|
||||||
"company": {
|
"company": {
|
||||||
"name": "Hoeger LLC",
|
"name": "Hoeger LLC",
|
||||||
"catchPhrase": "Centralized empowering task-force",
|
"catchPhrase": "Centralized empowering task-force",
|
||||||
"bs": "target end-to-end models"
|
"bs": "target end-to-end tables"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
|
@ -16,7 +16,7 @@ context("Create a automation", () => {
|
||||||
cy.contains("automate").click()
|
cy.contains("automate").click()
|
||||||
cy.contains("Create New Automation").click()
|
cy.contains("Create New Automation").click()
|
||||||
cy.get(".modal").within(() => {
|
cy.get(".modal").within(() => {
|
||||||
cy.get("input").type("Add Record")
|
cy.get("input").type("Add Row")
|
||||||
cy.get(".buttons")
|
cy.get(".buttons")
|
||||||
.contains("Create")
|
.contains("Create")
|
||||||
.click()
|
.click()
|
||||||
|
@ -24,7 +24,7 @@ context("Create a automation", () => {
|
||||||
|
|
||||||
// Add trigger
|
// Add trigger
|
||||||
cy.get("[data-cy=add-automation-component]").click()
|
cy.get("[data-cy=add-automation-component]").click()
|
||||||
cy.get("[data-cy=RECORD_SAVED]").click()
|
cy.get("[data-cy=ROW_SAVED]").click()
|
||||||
cy.get("[data-cy=automation-block-setup]").within(() => {
|
cy.get("[data-cy=automation-block-setup]").within(() => {
|
||||||
cy.get("select")
|
cy.get("select")
|
||||||
.first()
|
.first()
|
||||||
|
@ -32,7 +32,7 @@ context("Create a automation", () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
// Create action
|
// Create action
|
||||||
cy.get("[data-cy=CREATE_RECORD]").click()
|
cy.get("[data-cy=CREATE_ROW]").click()
|
||||||
cy.get("[data-cy=automation-block-setup]").within(() => {
|
cy.get("[data-cy=automation-block-setup]").within(() => {
|
||||||
cy.get("select")
|
cy.get("select")
|
||||||
.first()
|
.first()
|
||||||
|
@ -53,9 +53,9 @@ context("Create a automation", () => {
|
||||||
cy.get(".stop-button.highlighted").should("be.visible")
|
cy.get(".stop-button.highlighted").should("be.visible")
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should add record when a new record is added", () => {
|
it("should add row when a new row is added", () => {
|
||||||
cy.contains("backend").click()
|
cy.contains("backend").click()
|
||||||
cy.addRecord(["Rover", 15])
|
cy.addRow(["Rover", 15])
|
||||||
cy.reload()
|
cy.reload()
|
||||||
cy.contains("goodboy").should("have.text", "goodboy")
|
cy.contains("goodboy").should("have.text", "goodboy")
|
||||||
})
|
})
|
||||||
|
|
|
@ -4,9 +4,9 @@ xcontext('Create Components', () => {
|
||||||
cy.server()
|
cy.server()
|
||||||
cy.visit('localhost:4001/_builder')
|
cy.visit('localhost:4001/_builder')
|
||||||
// https://on.cypress.io/type
|
// https://on.cypress.io/type
|
||||||
cy.createApp('Model App', 'Model App Description')
|
cy.createApp('Table App', 'Table App Description')
|
||||||
cy.createTable('dog', 'name', 'age')
|
cy.createTable('dog', 'name', 'age')
|
||||||
cy.addRecord('bob', '15')
|
cy.addRow('bob', '15')
|
||||||
})
|
})
|
||||||
|
|
||||||
// https://on.cypress.io/interacting-with-elements
|
// https://on.cypress.io/interacting-with-elements
|
||||||
|
|
|
@ -16,8 +16,8 @@ context("Create a Table", () => {
|
||||||
cy.contains("name").should("be.visible")
|
cy.contains("name").should("be.visible")
|
||||||
})
|
})
|
||||||
|
|
||||||
it("creates a record in the table", () => {
|
it("creates a row in the table", () => {
|
||||||
cy.addRecord(["Rover"])
|
cy.addRow(["Rover"])
|
||||||
cy.contains("Rover").should("be.visible")
|
cy.contains("Rover").should("be.visible")
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -32,7 +32,7 @@ context("Create a Table", () => {
|
||||||
cy.contains("nameupdated").should("have.text", "nameupdated")
|
cy.contains("nameupdated").should("have.text", "nameupdated")
|
||||||
})
|
})
|
||||||
|
|
||||||
it("edits a record", () => {
|
it("edits a row", () => {
|
||||||
cy.get("tbody .ri-more-line").click()
|
cy.get("tbody .ri-more-line").click()
|
||||||
cy.get("[data-cy=edit-row]").click()
|
cy.get("[data-cy=edit-row]").click()
|
||||||
cy.get(".modal input").type("Updated")
|
cy.get(".modal input").type("Updated")
|
||||||
|
@ -40,7 +40,7 @@ context("Create a Table", () => {
|
||||||
cy.contains("RoverUpdated").should("have.text", "RoverUpdated")
|
cy.contains("RoverUpdated").should("have.text", "RoverUpdated")
|
||||||
})
|
})
|
||||||
|
|
||||||
it("deletes a record", () => {
|
it("deletes a row", () => {
|
||||||
cy.get("tbody .ri-more-line").click()
|
cy.get("tbody .ri-more-line").click()
|
||||||
cy.get("[data-cy=delete-row]").click()
|
cy.get("[data-cy=delete-row]").click()
|
||||||
cy.contains("Delete Row").click()
|
cy.contains("Delete Row").click()
|
||||||
|
|
|
@ -7,13 +7,13 @@ context("Create a View", () => {
|
||||||
cy.addColumn("data", "age", "Number")
|
cy.addColumn("data", "age", "Number")
|
||||||
cy.addColumn("data", "rating", "Number")
|
cy.addColumn("data", "rating", "Number")
|
||||||
|
|
||||||
// 6 Records
|
// 6 Rows
|
||||||
cy.addRecord(["Students", 25, 1])
|
cy.addRow(["Students", 25, 1])
|
||||||
cy.addRecord(["Students", 20, 3])
|
cy.addRow(["Students", 20, 3])
|
||||||
cy.addRecord(["Students", 18, 6])
|
cy.addRow(["Students", 18, 6])
|
||||||
cy.addRecord(["Students", 25, 2])
|
cy.addRow(["Students", 25, 2])
|
||||||
cy.addRecord(["Teachers", 49, 5])
|
cy.addRow(["Teachers", 49, 5])
|
||||||
cy.addRecord(["Teachers", 36, 3])
|
cy.addRow(["Teachers", 36, 3])
|
||||||
})
|
})
|
||||||
|
|
||||||
it("creates a view", () => {
|
it("creates a view", () => {
|
||||||
|
@ -109,7 +109,7 @@ context("Create a View", () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
it("renames a view", () => {
|
it("renames a view", () => {
|
||||||
cy.contains("[data-cy=model-nav-item]", "Test View")
|
cy.contains("[data-cy=table-nav-item]", "Test View")
|
||||||
.find(".ri-more-line")
|
.find(".ri-more-line")
|
||||||
.click()
|
.click()
|
||||||
cy.contains("Edit").click()
|
cy.contains("Edit").click()
|
||||||
|
@ -121,8 +121,8 @@ context("Create a View", () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
it("deletes a view", () => {
|
it("deletes a view", () => {
|
||||||
cy.contains("[data-cy=model-nav-item]", "Test View Updated").click()
|
cy.contains("[data-cy=table-nav-item]", "Test View Updated").click()
|
||||||
cy.contains("[data-cy=model-nav-item]", "Test View Updated")
|
cy.contains("[data-cy=table-nav-item]", "Test View Updated")
|
||||||
.find(".ri-more-line")
|
.find(".ri-more-line")
|
||||||
.click()
|
.click()
|
||||||
cy.contains("Delete").click()
|
cy.contains("Delete").click()
|
||||||
|
|
|
@ -3,7 +3,7 @@ context('Screen Tests', () => {
|
||||||
before(() => {
|
before(() => {
|
||||||
cy.server()
|
cy.server()
|
||||||
cy.visit('localhost:4001/_builder')
|
cy.visit('localhost:4001/_builder')
|
||||||
cy.createApp('Conor Cy App', 'Model App Description')
|
cy.createApp('Conor Cy App', 'Table App Description')
|
||||||
cy.navigateToFrontend()
|
cy.navigateToFrontend()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -64,7 +64,7 @@ Cypress.Commands.add("createTestTableWithData", () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
Cypress.Commands.add("createTable", tableName => {
|
Cypress.Commands.add("createTable", tableName => {
|
||||||
// Enter model name
|
// Enter table name
|
||||||
cy.contains("Create New Table").click()
|
cy.contains("Create New Table").click()
|
||||||
cy.get(".modal").within(() => {
|
cy.get(".modal").within(() => {
|
||||||
cy.get("input")
|
cy.get("input")
|
||||||
|
@ -92,7 +92,7 @@ Cypress.Commands.add("addColumn", (tableName, columnName, type) => {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
Cypress.Commands.add("addRecord", values => {
|
Cypress.Commands.add("addRow", values => {
|
||||||
cy.contains("Create New Row").click()
|
cy.contains("Create New Row").click()
|
||||||
|
|
||||||
cy.get(".modal").within(() => {
|
cy.get(".modal").within(() => {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "@budibase/builder",
|
"name": "@budibase/builder",
|
||||||
"version": "0.2.0",
|
"version": "0.2.1",
|
||||||
"license": "AGPL-3.0",
|
"license": "AGPL-3.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
@ -64,7 +64,7 @@
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@budibase/bbui": "^1.41.0",
|
"@budibase/bbui": "^1.41.0",
|
||||||
"@budibase/client": "^0.2.0",
|
"@budibase/client": "^0.2.1",
|
||||||
"@budibase/colorpicker": "^1.0.1",
|
"@budibase/colorpicker": "^1.0.1",
|
||||||
"@fortawesome/fontawesome-free": "^5.14.0",
|
"@fortawesome/fontawesome-free": "^5.14.0",
|
||||||
"@sentry/browser": "5.19.1",
|
"@sentry/browser": "5.19.1",
|
||||||
|
|
|
@ -6,7 +6,7 @@ import { cloneDeep, difference } from "lodash/fp"
|
||||||
* @property {string} componentInstanceId - an _id of a component that has been added to a screen, which you want to fetch bindable props for
|
* @property {string} componentInstanceId - an _id of a component that has been added to a screen, which you want to fetch bindable props for
|
||||||
* @propperty {Object} screen - current screen - where componentInstanceId lives
|
* @propperty {Object} screen - current screen - where componentInstanceId lives
|
||||||
* @property {Object} components - dictionary of component definitions
|
* @property {Object} components - dictionary of component definitions
|
||||||
* @property {Array} models - array of all models
|
* @property {Array} tables - array of all tables
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -23,13 +23,13 @@ import { cloneDeep, difference } from "lodash/fp"
|
||||||
* @param {fetchBindablePropertiesParameter} param
|
* @param {fetchBindablePropertiesParameter} param
|
||||||
* @returns {Array.<BindableProperty>}
|
* @returns {Array.<BindableProperty>}
|
||||||
*/
|
*/
|
||||||
export default function({ componentInstanceId, screen, components, models }) {
|
export default function({ componentInstanceId, screen, components, tables }) {
|
||||||
const walkResult = walk({
|
const walkResult = walk({
|
||||||
// cloning so we are free to mutate props (e.g. by adding _contexts)
|
// cloning so we are free to mutate props (e.g. by adding _contexts)
|
||||||
instance: cloneDeep(screen.props),
|
instance: cloneDeep(screen.props),
|
||||||
targetId: componentInstanceId,
|
targetId: componentInstanceId,
|
||||||
components,
|
components,
|
||||||
models,
|
tables,
|
||||||
})
|
})
|
||||||
|
|
||||||
return [
|
return [
|
||||||
|
@ -38,7 +38,7 @@ export default function({ componentInstanceId, screen, components, models }) {
|
||||||
.map(componentInstanceToBindable(walkResult)),
|
.map(componentInstanceToBindable(walkResult)),
|
||||||
|
|
||||||
...(walkResult.target?._contexts
|
...(walkResult.target?._contexts
|
||||||
.map(contextToBindables(models, walkResult))
|
.map(contextToBindables(tables, walkResult))
|
||||||
.flat() ?? []),
|
.flat() ?? []),
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -71,14 +71,14 @@ const componentInstanceToBindable = walkResult => i => {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const contextToBindables = (models, walkResult) => context => {
|
const contextToBindables = (tables, walkResult) => context => {
|
||||||
const contextParentPath = getParentPath(walkResult, context)
|
const contextParentPath = getParentPath(walkResult, context)
|
||||||
const modelId = context.model?.modelId ?? context.model
|
const tableId = context.table?.tableId ?? context.table
|
||||||
const model = models.find(model => model._id === modelId)
|
const table = tables.find(table => table._id === tableId)
|
||||||
let schema =
|
let schema =
|
||||||
context.model?.type === "view"
|
context.table?.type === "view"
|
||||||
? model?.views?.[context.model.name]?.schema
|
? table?.views?.[context.table.name]?.schema
|
||||||
: model?.schema
|
: table?.schema
|
||||||
|
|
||||||
// Avoid crashing whenever no data source has been selected
|
// Avoid crashing whenever no data source has been selected
|
||||||
if (!schema) {
|
if (!schema) {
|
||||||
|
@ -98,9 +98,9 @@ const contextToBindables = (models, walkResult) => context => {
|
||||||
// how the binding expression persists, and is used in the app at runtime
|
// how the binding expression persists, and is used in the app at runtime
|
||||||
runtimeBinding: `${contextParentPath}data.${runtimeBoundKey}`,
|
runtimeBinding: `${contextParentPath}data.${runtimeBoundKey}`,
|
||||||
// how the binding expressions looks to the user of the builder
|
// how the binding expressions looks to the user of the builder
|
||||||
readableBinding: `${context.instance._instanceName}.${model.name}.${key}`,
|
readableBinding: `${context.instance._instanceName}.${table.name}.${key}`,
|
||||||
// model / view info
|
// table / view info
|
||||||
model: context.model,
|
table: context.table,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -126,7 +126,7 @@ const getParentPath = (walkResult, context) => {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
const walk = ({ instance, targetId, components, models, result }) => {
|
const walk = ({ instance, targetId, components, tables, result }) => {
|
||||||
if (!result) {
|
if (!result) {
|
||||||
result = {
|
result = {
|
||||||
target: null,
|
target: null,
|
||||||
|
@ -165,8 +165,8 @@ const walk = ({ instance, targetId, components, models, result }) => {
|
||||||
if (contextualInstance) {
|
if (contextualInstance) {
|
||||||
// add to currentContexts (ancestory of context)
|
// add to currentContexts (ancestory of context)
|
||||||
// before walking children
|
// before walking children
|
||||||
const model = instance[component.context]
|
const table = instance[component.context]
|
||||||
result.currentContexts.push({ instance, model })
|
result.currentContexts.push({ instance, table })
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentContexts = [...result.currentContexts]
|
const currentContexts = [...result.currentContexts]
|
||||||
|
@ -175,7 +175,7 @@ const walk = ({ instance, targetId, components, models, result }) => {
|
||||||
// these have been deep cloned above, so shouln't modify the
|
// these have been deep cloned above, so shouln't modify the
|
||||||
// original component instances
|
// original component instances
|
||||||
child._contexts = currentContexts
|
child._contexts = currentContexts
|
||||||
walk({ instance: child, targetId, components, models, result })
|
walk({ instance: child, targetId, components, tables, result })
|
||||||
}
|
}
|
||||||
|
|
||||||
if (contextualInstance) {
|
if (contextualInstance) {
|
||||||
|
|
|
@ -26,16 +26,16 @@ export default {
|
||||||
],
|
],
|
||||||
trigger: {
|
trigger: {
|
||||||
id: "iRzYMOqND",
|
id: "iRzYMOqND",
|
||||||
name: "Record Saved",
|
name: "Row Saved",
|
||||||
event: "record:save",
|
event: "row:save",
|
||||||
icon: "ri-save-line",
|
icon: "ri-save-line",
|
||||||
tagline: "Record is added to <b>{{model.name}}</b>",
|
tagline: "Row is added to <b>{{table.name}}</b>",
|
||||||
description: "Fired when a record is saved to your database.",
|
description: "Fired when a row is saved to your database.",
|
||||||
params: { model: "model" },
|
params: { table: "table" },
|
||||||
type: "TRIGGER",
|
type: "TRIGGER",
|
||||||
args: {
|
args: {
|
||||||
model: {
|
table: {
|
||||||
type: "model",
|
type: "table",
|
||||||
views: {},
|
views: {},
|
||||||
name: "users",
|
name: "users",
|
||||||
schema: {
|
schema: {
|
||||||
|
@ -65,7 +65,7 @@ export default {
|
||||||
_rev: "7-b8aa1ce0b53e88928bb88fc11bdc0aff",
|
_rev: "7-b8aa1ce0b53e88928bb88fc11bdc0aff",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
stepId: "RECORD_SAVED",
|
stepId: "ROW_SAVED",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
type: "automation",
|
type: "automation",
|
||||||
|
|
|
@ -3,12 +3,12 @@ import { cloneDeep } from "lodash/fp"
|
||||||
import api from "../api"
|
import api from "../api"
|
||||||
|
|
||||||
const INITIAL_BACKEND_UI_STATE = {
|
const INITIAL_BACKEND_UI_STATE = {
|
||||||
models: [],
|
tables: [],
|
||||||
views: [],
|
views: [],
|
||||||
users: [],
|
users: [],
|
||||||
selectedDatabase: {},
|
selectedDatabase: {},
|
||||||
selectedModel: {},
|
selectedTable: {},
|
||||||
draftModel: {},
|
draftTable: {},
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getBackendUiStore = () => {
|
export const getBackendUiStore = () => {
|
||||||
|
@ -18,16 +18,16 @@ export const getBackendUiStore = () => {
|
||||||
reset: () => store.set({ ...INITIAL_BACKEND_UI_STATE }),
|
reset: () => store.set({ ...INITIAL_BACKEND_UI_STATE }),
|
||||||
database: {
|
database: {
|
||||||
select: async db => {
|
select: async db => {
|
||||||
const modelsResponse = await api.get(`/api/models`)
|
const tablesResponse = await api.get(`/api/tables`)
|
||||||
const models = await modelsResponse.json()
|
const tables = await tablesResponse.json()
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
state.selectedDatabase = db
|
state.selectedDatabase = db
|
||||||
state.models = models
|
state.tables = tables
|
||||||
return state
|
return state
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
records: {
|
rows: {
|
||||||
save: () =>
|
save: () =>
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
state.selectedView = state.selectedView
|
state.selectedView = state.selectedView
|
||||||
|
@ -38,56 +38,56 @@ export const getBackendUiStore = () => {
|
||||||
state.selectedView = state.selectedView
|
state.selectedView = state.selectedView
|
||||||
return state
|
return state
|
||||||
}),
|
}),
|
||||||
select: record =>
|
select: row =>
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
state.selectedRecord = record
|
state.selectedRow = row
|
||||||
return state
|
return state
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
models: {
|
tables: {
|
||||||
fetch: async () => {
|
fetch: async () => {
|
||||||
const modelsResponse = await api.get(`/api/models`)
|
const tablesResponse = await api.get(`/api/tables`)
|
||||||
const models = await modelsResponse.json()
|
const tables = await tablesResponse.json()
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
state.models = models
|
state.tables = tables
|
||||||
return state
|
return state
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
select: model =>
|
select: table =>
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
state.selectedModel = model
|
state.selectedTable = table
|
||||||
state.draftModel = cloneDeep(model)
|
state.draftTable = cloneDeep(table)
|
||||||
state.selectedView = { name: `all_${model._id}` }
|
state.selectedView = { name: `all_${table._id}` }
|
||||||
return state
|
return state
|
||||||
}),
|
}),
|
||||||
save: async model => {
|
save: async table => {
|
||||||
const updatedModel = cloneDeep(model)
|
const updatedTable = cloneDeep(table)
|
||||||
|
|
||||||
// update any renamed schema keys to reflect their names
|
// update any renamed schema keys to reflect their names
|
||||||
for (let key in updatedModel.schema) {
|
for (let key in updatedTable.schema) {
|
||||||
const field = updatedModel.schema[key]
|
const field = updatedTable.schema[key]
|
||||||
// field has been renamed
|
// field has been renamed
|
||||||
if (field.name && field.name !== key) {
|
if (field.name && field.name !== key) {
|
||||||
updatedModel.schema[field.name] = field
|
updatedTable.schema[field.name] = field
|
||||||
updatedModel._rename = { old: key, updated: field.name }
|
updatedTable._rename = { old: key, updated: field.name }
|
||||||
delete updatedModel.schema[key]
|
delete updatedTable.schema[key]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const SAVE_MODEL_URL = `/api/models`
|
const SAVE_TABLE_URL = `/api/tables`
|
||||||
const response = await api.post(SAVE_MODEL_URL, updatedModel)
|
const response = await api.post(SAVE_TABLE_URL, updatedTable)
|
||||||
const savedModel = await response.json()
|
const savedTable = await response.json()
|
||||||
await store.actions.models.fetch()
|
await store.actions.tables.fetch()
|
||||||
store.actions.models.select(savedModel)
|
store.actions.tables.select(savedTable)
|
||||||
return savedModel
|
return savedTable
|
||||||
},
|
},
|
||||||
delete: async model => {
|
delete: async table => {
|
||||||
await api.delete(`/api/models/${model._id}/${model._rev}`)
|
await api.delete(`/api/tables/${table._id}/${table._rev}`)
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
state.models = state.models.filter(
|
state.tables = state.tables.filter(
|
||||||
existing => existing._id !== model._id
|
existing => existing._id !== table._id
|
||||||
)
|
)
|
||||||
state.selectedModel = {}
|
state.selectedTable = {}
|
||||||
return state
|
return state
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
@ -95,23 +95,23 @@ export const getBackendUiStore = () => {
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
// delete the original if renaming
|
// delete the original if renaming
|
||||||
if (originalName) {
|
if (originalName) {
|
||||||
delete state.draftModel.schema[originalName]
|
delete state.draftTable.schema[originalName]
|
||||||
state.draftModel._rename = {
|
state.draftTable._rename = {
|
||||||
old: originalName,
|
old: originalName,
|
||||||
updated: field.name,
|
updated: field.name,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
state.draftModel.schema[field.name] = cloneDeep(field)
|
state.draftTable.schema[field.name] = cloneDeep(field)
|
||||||
|
|
||||||
store.actions.models.save(state.draftModel)
|
store.actions.tables.save(state.draftTable)
|
||||||
return state
|
return state
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
deleteField: field => {
|
deleteField: field => {
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
delete state.draftModel.schema[field.name]
|
delete state.draftTable.schema[field.name]
|
||||||
store.actions.models.save(state.draftModel)
|
store.actions.tables.save(state.draftTable)
|
||||||
return state
|
return state
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
@ -120,12 +120,12 @@ export const getBackendUiStore = () => {
|
||||||
select: view =>
|
select: view =>
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
state.selectedView = view
|
state.selectedView = view
|
||||||
state.selectedModel = {}
|
state.selectedTable = {}
|
||||||
return state
|
return state
|
||||||
}),
|
}),
|
||||||
delete: async view => {
|
delete: async view => {
|
||||||
await api.delete(`/api/views/${view}`)
|
await api.delete(`/api/views/${view}`)
|
||||||
await store.actions.models.fetch()
|
await store.actions.tables.fetch()
|
||||||
},
|
},
|
||||||
save: async view => {
|
save: async view => {
|
||||||
const response = await api.post(`/api/views`, view)
|
const response = await api.post(`/api/views`, view)
|
||||||
|
@ -137,14 +137,14 @@ export const getBackendUiStore = () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
const viewModel = state.models.find(
|
const viewTable = state.tables.find(
|
||||||
model => model._id === view.modelId
|
table => table._id === view.tableId
|
||||||
)
|
)
|
||||||
|
|
||||||
if (view.originalName) delete viewModel.views[view.originalName]
|
if (view.originalName) delete viewTable.views[view.originalName]
|
||||||
viewModel.views[view.name] = viewMeta
|
viewTable.views[view.name] = viewMeta
|
||||||
|
|
||||||
state.models = state.models
|
state.tables = state.tables
|
||||||
state.selectedView = viewMeta
|
state.selectedView = viewMeta
|
||||||
return state
|
return state
|
||||||
})
|
})
|
||||||
|
|
|
@ -15,7 +15,7 @@ const createScreen = () => ({
|
||||||
},
|
},
|
||||||
_children: [],
|
_children: [],
|
||||||
_instanceName: "",
|
_instanceName: "",
|
||||||
model: "",
|
table: "",
|
||||||
},
|
},
|
||||||
route: "",
|
route: "",
|
||||||
name: "screen-id",
|
name: "screen-id",
|
|
@ -15,7 +15,7 @@ const createScreen = () => ({
|
||||||
},
|
},
|
||||||
_children: [],
|
_children: [],
|
||||||
_instanceName: "",
|
_instanceName: "",
|
||||||
model: "",
|
table: "",
|
||||||
},
|
},
|
||||||
route: "",
|
route: "",
|
||||||
name: "screen-id",
|
name: "screen-id",
|
|
@ -1,19 +1,19 @@
|
||||||
import newRecordScreen from "./newRecordScreen"
|
import newRowScreen from "./newRowScreen"
|
||||||
import recordDetailScreen from "./recordDetailScreen"
|
import rowDetailScreen from "./rowDetailScreen"
|
||||||
import recordListScreen from "./recordListScreen"
|
import rowListScreen from "./rowListScreen"
|
||||||
import emptyNewRecordScreen from "./emptyNewRecordScreen"
|
import emptyNewRowScreen from "./emptyNewRowScreen"
|
||||||
import createFromScratchScreen from "./createFromScratchScreen"
|
import createFromScratchScreen from "./createFromScratchScreen"
|
||||||
import emptyRecordDetailScreen from "./emptyRecordDetailScreen"
|
import emptyRowDetailScreen from "./emptyRowDetailScreen"
|
||||||
import { generateNewIdsForComponent } from "../../storeUtils"
|
import { generateNewIdsForComponent } from "../../storeUtils"
|
||||||
import { uuid } from "builderStore/uuid"
|
import { uuid } from "builderStore/uuid"
|
||||||
|
|
||||||
const allTemplates = models => [
|
const allTemplates = tables => [
|
||||||
createFromScratchScreen,
|
createFromScratchScreen,
|
||||||
...newRecordScreen(models),
|
...newRowScreen(tables),
|
||||||
...recordDetailScreen(models),
|
...rowDetailScreen(tables),
|
||||||
...recordListScreen(models),
|
...rowListScreen(tables),
|
||||||
emptyNewRecordScreen,
|
emptyNewRowScreen,
|
||||||
emptyRecordDetailScreen,
|
emptyRowDetailScreen,
|
||||||
]
|
]
|
||||||
|
|
||||||
// allows us to apply common behaviour to all create() functions
|
// allows us to apply common behaviour to all create() functions
|
||||||
|
@ -28,8 +28,8 @@ const createTemplateOverride = (frontendState, create) => () => {
|
||||||
return screen
|
return screen
|
||||||
}
|
}
|
||||||
|
|
||||||
export default (frontendState, models) =>
|
export default (frontendState, tables) =>
|
||||||
allTemplates(models).map(template => ({
|
allTemplates(tables).map(template => ({
|
||||||
...template,
|
...template,
|
||||||
create: createTemplateOverride(frontendState, template.create),
|
create: createTemplateOverride(frontendState, template.create),
|
||||||
}))
|
}))
|
||||||
|
|
|
@ -1,18 +1,18 @@
|
||||||
export default function(models) {
|
export default function(tables) {
|
||||||
return models.map(model => {
|
return tables.map(table => {
|
||||||
const fields = Object.keys(model.schema)
|
const fields = Object.keys(table.schema)
|
||||||
const heading = fields.length > 0 ? `{{ data.${fields[0]} }}` : "Add Row"
|
const heading = fields.length > 0 ? `{{ data.${fields[0]} }}` : "Add Row"
|
||||||
return {
|
return {
|
||||||
name: `${model.name} - New`,
|
name: `${table.name} - New`,
|
||||||
create: () => createScreen(model, heading),
|
create: () => createScreen(table, heading),
|
||||||
id: NEW_RECORD_TEMPLATE,
|
id: NEW_ROW_TEMPLATE,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export const NEW_RECORD_TEMPLATE = "NEW_RECORD_TEMPLATE"
|
export const NEW_ROW_TEMPLATE = "NEW_ROW_TEMPLATE"
|
||||||
|
|
||||||
const createScreen = (model, heading) => ({
|
const createScreen = (table, heading) => ({
|
||||||
props: {
|
props: {
|
||||||
_id: "",
|
_id: "",
|
||||||
_component: "@budibase/standard-components/newrow",
|
_component: "@budibase/standard-components/newrow",
|
||||||
|
@ -22,7 +22,7 @@ const createScreen = (model, heading) => ({
|
||||||
active: {},
|
active: {},
|
||||||
selected: {},
|
selected: {},
|
||||||
},
|
},
|
||||||
model: model._id,
|
table: table._id,
|
||||||
_children: [
|
_children: [
|
||||||
{
|
{
|
||||||
_id: "",
|
_id: "",
|
||||||
|
@ -50,7 +50,7 @@ const createScreen = (model, heading) => ({
|
||||||
selected: {},
|
selected: {},
|
||||||
},
|
},
|
||||||
_code: "",
|
_code: "",
|
||||||
_instanceName: `${model.name} Form`,
|
_instanceName: `${table.name} Form`,
|
||||||
_children: [],
|
_children: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -91,7 +91,7 @@ const createScreen = (model, heading) => ({
|
||||||
onClick: [
|
onClick: [
|
||||||
{
|
{
|
||||||
parameters: {
|
parameters: {
|
||||||
url: `/${model.name.toLowerCase()}`,
|
url: `/${table.name.toLowerCase()}`,
|
||||||
},
|
},
|
||||||
"##eventHandlerType": "Navigate To",
|
"##eventHandlerType": "Navigate To",
|
||||||
},
|
},
|
||||||
|
@ -116,9 +116,9 @@ const createScreen = (model, heading) => ({
|
||||||
{
|
{
|
||||||
parameters: {
|
parameters: {
|
||||||
contextPath: "data",
|
contextPath: "data",
|
||||||
modelId: model._id,
|
tableId: table._id,
|
||||||
},
|
},
|
||||||
"##eventHandlerType": "Save Record",
|
"##eventHandlerType": "Save Row",
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
_instanceName: "Save Button",
|
_instanceName: "Save Button",
|
||||||
|
@ -127,9 +127,9 @@ const createScreen = (model, heading) => ({
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
_instanceName: `${model.name} - New`,
|
_instanceName: `${table.name} - New`,
|
||||||
_code: "",
|
_code: "",
|
||||||
},
|
},
|
||||||
route: `/${model.name.toLowerCase()}/new`,
|
route: `/${table.name.toLowerCase()}/new`,
|
||||||
name: "",
|
name: "",
|
||||||
})
|
})
|
|
@ -1,18 +1,18 @@
|
||||||
export default function(models) {
|
export default function(tables) {
|
||||||
return models.map(model => {
|
return tables.map(table => {
|
||||||
const fields = Object.keys(model.schema)
|
const fields = Object.keys(table.schema)
|
||||||
const heading = fields.length > 0 ? `{{ data.${fields[0]} }}` : "Detail"
|
const heading = fields.length > 0 ? `{{ data.${fields[0]} }}` : "Detail"
|
||||||
return {
|
return {
|
||||||
name: `${model.name} - Detail`,
|
name: `${table.name} - Detail`,
|
||||||
create: () => createScreen(model, heading),
|
create: () => createScreen(table, heading),
|
||||||
id: RECORD_DETAIL_TEMPLATE,
|
id: ROW_DETAIL_TEMPLATE,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export const RECORD_DETAIL_TEMPLATE = "RECORD_DETAIL_TEMPLATE"
|
export const ROW_DETAIL_TEMPLATE = "ROW_DETAIL_TEMPLATE"
|
||||||
|
|
||||||
const createScreen = (model, heading) => ({
|
const createScreen = (table, heading) => ({
|
||||||
props: {
|
props: {
|
||||||
_id: "",
|
_id: "",
|
||||||
_component: "@budibase/standard-components/rowdetail",
|
_component: "@budibase/standard-components/rowdetail",
|
||||||
|
@ -22,7 +22,7 @@ const createScreen = (model, heading) => ({
|
||||||
active: {},
|
active: {},
|
||||||
selected: {},
|
selected: {},
|
||||||
},
|
},
|
||||||
model: model._id,
|
table: table._id,
|
||||||
_children: [
|
_children: [
|
||||||
{
|
{
|
||||||
_id: "",
|
_id: "",
|
||||||
|
@ -50,7 +50,7 @@ const createScreen = (model, heading) => ({
|
||||||
selected: {},
|
selected: {},
|
||||||
},
|
},
|
||||||
_code: "",
|
_code: "",
|
||||||
_instanceName: `${model.name} Form`,
|
_instanceName: `${table.name} Form`,
|
||||||
_children: [],
|
_children: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -91,7 +91,7 @@ const createScreen = (model, heading) => ({
|
||||||
onClick: [
|
onClick: [
|
||||||
{
|
{
|
||||||
parameters: {
|
parameters: {
|
||||||
url: `/${model.name.toLowerCase()}`,
|
url: `/${table.name.toLowerCase()}`,
|
||||||
},
|
},
|
||||||
"##eventHandlerType": "Navigate To",
|
"##eventHandlerType": "Navigate To",
|
||||||
},
|
},
|
||||||
|
@ -116,9 +116,9 @@ const createScreen = (model, heading) => ({
|
||||||
{
|
{
|
||||||
parameters: {
|
parameters: {
|
||||||
contextPath: "data",
|
contextPath: "data",
|
||||||
modelId: model._id,
|
tableId: table._id,
|
||||||
},
|
},
|
||||||
"##eventHandlerType": "Save Record",
|
"##eventHandlerType": "Save Row",
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
_instanceName: "Save Button",
|
_instanceName: "Save Button",
|
||||||
|
@ -127,9 +127,9 @@ const createScreen = (model, heading) => ({
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
_instanceName: `${model.name} - Detail`,
|
_instanceName: `${table.name} - Detail`,
|
||||||
_code: "",
|
_code: "",
|
||||||
},
|
},
|
||||||
route: `/${model.name.toLowerCase()}/:id`,
|
route: `/${table.name.toLowerCase()}/:id`,
|
||||||
name: "",
|
name: "",
|
||||||
})
|
})
|
|
@ -1,16 +1,16 @@
|
||||||
export default function(models) {
|
export default function(tables) {
|
||||||
return models.map(model => {
|
return tables.map(table => {
|
||||||
return {
|
return {
|
||||||
name: `${model.name} - List`,
|
name: `${table.name} - List`,
|
||||||
create: () => createScreen(model),
|
create: () => createScreen(table),
|
||||||
id: RECORD_LIST_TEMPLATE,
|
id: ROW_LIST_TEMPLATE,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export const RECORD_LIST_TEMPLATE = "RECORD_LIST_TEMPLATE"
|
export const ROW_LIST_TEMPLATE = "ROW_LIST_TEMPLATE"
|
||||||
|
|
||||||
const createScreen = model => ({
|
const createScreen = table => ({
|
||||||
props: {
|
props: {
|
||||||
_id: "",
|
_id: "",
|
||||||
_component: "@budibase/standard-components/container",
|
_component: "@budibase/standard-components/container",
|
||||||
|
@ -53,7 +53,7 @@ const createScreen = model => ({
|
||||||
},
|
},
|
||||||
_code: "",
|
_code: "",
|
||||||
className: "",
|
className: "",
|
||||||
text: `${model.name} List`,
|
text: `${table.name} List`,
|
||||||
type: "h1",
|
type: "h1",
|
||||||
_instanceName: "Heading 1",
|
_instanceName: "Heading 1",
|
||||||
_children: [],
|
_children: [],
|
||||||
|
@ -74,7 +74,7 @@ const createScreen = model => ({
|
||||||
onClick: [
|
onClick: [
|
||||||
{
|
{
|
||||||
parameters: {
|
parameters: {
|
||||||
url: `/${model.name}/new`,
|
url: `/${table.name}/new`,
|
||||||
},
|
},
|
||||||
"##eventHandlerType": "Navigate To",
|
"##eventHandlerType": "Navigate To",
|
||||||
},
|
},
|
||||||
|
@ -96,19 +96,19 @@ const createScreen = model => ({
|
||||||
_code: "",
|
_code: "",
|
||||||
datasource: {
|
datasource: {
|
||||||
label: "Deals",
|
label: "Deals",
|
||||||
name: `all_${model._id}`,
|
name: `all_${table._id}`,
|
||||||
modelId: model._id,
|
tableId: table._id,
|
||||||
type: "model",
|
type: "table",
|
||||||
},
|
},
|
||||||
_instanceName: `${model.name} Table`,
|
_instanceName: `${table.name} Table`,
|
||||||
_children: [],
|
_children: [],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
_instanceName: `${model.name} - List`,
|
_instanceName: `${table.name} - List`,
|
||||||
_code: "",
|
_code: "",
|
||||||
className: "",
|
className: "",
|
||||||
onLoad: [],
|
onLoad: [],
|
||||||
},
|
},
|
||||||
route: `/${model.name.toLowerCase()}`,
|
route: `/${table.name.toLowerCase()}`,
|
||||||
name: "",
|
name: "",
|
||||||
})
|
})
|
|
@ -13,10 +13,10 @@
|
||||||
|
|
||||||
function enrichInputs(inputs) {
|
function enrichInputs(inputs) {
|
||||||
let enrichedInputs = { ...inputs, enriched: {} }
|
let enrichedInputs = { ...inputs, enriched: {} }
|
||||||
const modelId = inputs.modelId || inputs.record?.modelId
|
const tableId = inputs.tableId || inputs.row?.tableId
|
||||||
if (modelId) {
|
if (tableId) {
|
||||||
enrichedInputs.enriched.model = $backendUiStore.models.find(
|
enrichedInputs.enriched.table = $backendUiStore.tables.find(
|
||||||
model => model._id === modelId
|
table => table._id === tableId
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
return enrichedInputs
|
return enrichedInputs
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<script>
|
<script>
|
||||||
import ModelSelector from "./ParamInputs/ModelSelector.svelte"
|
import TableSelector from "./ParamInputs/TableSelector.svelte"
|
||||||
import RecordSelector from "./ParamInputs/RecordSelector.svelte"
|
import RowSelector from "./ParamInputs/RowSelector.svelte"
|
||||||
import { Input, TextArea, Select, Label } from "@budibase/bbui"
|
import { Input, TextArea, Select, Label } from "@budibase/bbui"
|
||||||
import { automationStore } from "builderStore"
|
import { automationStore } from "builderStore"
|
||||||
import BindableInput from "../../userInterface/BindableInput.svelte"
|
import BindableInput from "../../userInterface/BindableInput.svelte"
|
||||||
|
@ -60,10 +60,10 @@
|
||||||
</Select>
|
</Select>
|
||||||
{:else if value.customType === 'password'}
|
{:else if value.customType === 'password'}
|
||||||
<Input type="password" thin bind:value={block.inputs[key]} />
|
<Input type="password" thin bind:value={block.inputs[key]} />
|
||||||
{:else if value.customType === 'model'}
|
{:else if value.customType === 'table'}
|
||||||
<ModelSelector bind:value={block.inputs[key]} />
|
<TableSelector bind:value={block.inputs[key]} />
|
||||||
{:else if value.customType === 'record'}
|
{:else if value.customType === 'row'}
|
||||||
<RecordSelector bind:value={block.inputs[key]} {bindings} />
|
<RowSelector bind:value={block.inputs[key]} {bindings} />
|
||||||
{:else if value.type === 'string' || value.type === 'number'}
|
{:else if value.type === 'string' || value.type === 'number'}
|
||||||
<BindableInput
|
<BindableInput
|
||||||
type="string"
|
type="string"
|
||||||
|
|
|
@ -6,12 +6,12 @@
|
||||||
export let value
|
export let value
|
||||||
export let bindings
|
export let bindings
|
||||||
|
|
||||||
$: model = $backendUiStore.models.find(model => model._id === value?.modelId)
|
$: table = $backendUiStore.tables.find(table => table._id === value?.tableId)
|
||||||
$: schemaFields = Object.entries(model?.schema ?? {})
|
$: schemaFields = Object.entries(table?.schema ?? {})
|
||||||
|
|
||||||
// Ensure any nullish modelId values get set to empty string so
|
// Ensure any nullish tableId values get set to empty string so
|
||||||
// that the select works
|
// that the select works
|
||||||
$: if (value?.modelId == null) value = { modelId: "" }
|
$: if (value?.tableId == null) value = { tableId: "" }
|
||||||
|
|
||||||
function schemaHasOptions(schema) {
|
function schemaHasOptions(schema) {
|
||||||
return !!schema.constraints?.inclusion?.length
|
return !!schema.constraints?.inclusion?.length
|
||||||
|
@ -19,10 +19,10 @@
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="block-field">
|
<div class="block-field">
|
||||||
<Select bind:value={value.modelId} thin secondary>
|
<Select bind:value={value.tableId} thin secondary>
|
||||||
<option value="">Choose an option</option>
|
<option value="">Choose an option</option>
|
||||||
{#each $backendUiStore.models as model}
|
{#each $backendUiStore.tables as table}
|
||||||
<option value={model._id}>{model.name}</option>
|
<option value={table._id}>{table.name}</option>
|
||||||
{/each}
|
{/each}
|
||||||
</Select>
|
</Select>
|
||||||
</div>
|
</div>
|
|
@ -8,8 +8,8 @@
|
||||||
<div class="block-field">
|
<div class="block-field">
|
||||||
<Select bind:value secondary thin>
|
<Select bind:value secondary thin>
|
||||||
<option value="">Choose an option</option>
|
<option value="">Choose an option</option>
|
||||||
{#each $backendUiStore.models as model}
|
{#each $backendUiStore.tables as table}
|
||||||
<option value={model._id}>{model.name}</option>
|
<option value={table._id}>{table.name}</option>
|
||||||
{/each}
|
{/each}
|
||||||
</Select>
|
</Select>
|
||||||
</div>
|
</div>
|
|
@ -10,19 +10,19 @@
|
||||||
let data = []
|
let data = []
|
||||||
let loading = false
|
let loading = false
|
||||||
|
|
||||||
$: title = $backendUiStore.selectedModel.name
|
$: title = $backendUiStore.selectedTable.name
|
||||||
$: schema = $backendUiStore.selectedModel.schema
|
$: schema = $backendUiStore.selectedTable.schema
|
||||||
$: modelView = {
|
$: tableView = {
|
||||||
schema,
|
schema,
|
||||||
name: $backendUiStore.selectedView.name,
|
name: $backendUiStore.selectedView.name,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fetch records for specified model
|
// Fetch rows for specified table
|
||||||
$: {
|
$: {
|
||||||
if ($backendUiStore.selectedView?.name?.startsWith("all_")) {
|
if ($backendUiStore.selectedView?.name?.startsWith("all_")) {
|
||||||
loading = true
|
loading = true
|
||||||
api.fetchDataForView($backendUiStore.selectedView).then(records => {
|
api.fetchDataForView($backendUiStore.selectedView).then(rows => {
|
||||||
data = records || []
|
data = rows || []
|
||||||
loading = false
|
loading = false
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -34,6 +34,6 @@
|
||||||
{#if Object.keys(schema).length > 0}
|
{#if Object.keys(schema).length > 0}
|
||||||
<CreateRowButton />
|
<CreateRowButton />
|
||||||
<CreateViewButton />
|
<CreateViewButton />
|
||||||
<ExportButton view={modelView} />
|
<ExportButton view={tableView} />
|
||||||
{/if}
|
{/if}
|
||||||
</Table>
|
</Table>
|
|
@ -4,37 +4,37 @@
|
||||||
import { onMount } from "svelte"
|
import { onMount } from "svelte"
|
||||||
import { backendUiStore } from "builderStore"
|
import { backendUiStore } from "builderStore"
|
||||||
|
|
||||||
export let modelId
|
export let tableId
|
||||||
export let recordId
|
export let rowId
|
||||||
export let fieldName
|
export let fieldName
|
||||||
|
|
||||||
let record
|
let row
|
||||||
let title
|
let title
|
||||||
|
|
||||||
$: data = record?.[fieldName] ?? []
|
$: data = row?.[fieldName] ?? []
|
||||||
$: linkedModelId = data?.length ? data[0].modelId : null
|
$: linkedTableId = data?.length ? data[0].tableId : null
|
||||||
$: linkedModel = $backendUiStore.models.find(
|
$: linkedTable = $backendUiStore.tables.find(
|
||||||
model => model._id === linkedModelId
|
table => table._id === linkedTableId
|
||||||
)
|
)
|
||||||
$: schema = linkedModel?.schema
|
$: schema = linkedTable?.schema
|
||||||
$: model = $backendUiStore.models.find(model => model._id === modelId)
|
$: table = $backendUiStore.tables.find(table => table._id === tableId)
|
||||||
$: fetchData(modelId, recordId)
|
$: fetchData(tableId, rowId)
|
||||||
$: {
|
$: {
|
||||||
let recordLabel = record?.[model?.primaryDisplay]
|
let rowLabel = row?.[table?.primaryDisplay]
|
||||||
if (recordLabel) {
|
if (rowLabel) {
|
||||||
title = `${recordLabel} - ${fieldName}`
|
title = `${rowLabel} - ${fieldName}`
|
||||||
} else {
|
} else {
|
||||||
title = fieldName
|
title = fieldName
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function fetchData(modelId, recordId) {
|
async function fetchData(tableId, rowId) {
|
||||||
const QUERY_VIEW_URL = `/api/${modelId}/${recordId}/enrich`
|
const QUERY_VIEW_URL = `/api/${tableId}/${rowId}/enrich`
|
||||||
const response = await api.get(QUERY_VIEW_URL)
|
const response = await api.get(QUERY_VIEW_URL)
|
||||||
record = await response.json()
|
row = await response.json()
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if record && record._id === recordId}
|
{#if row && row._id === rowId}
|
||||||
<Table {title} {schema} {data} />
|
<Table {title} {schema} {data} />
|
||||||
{/if}
|
{/if}
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
import { Input, Select, Label, DatePicker, Toggle } from "@budibase/bbui"
|
import { Input, Select, Label, DatePicker, Toggle } from "@budibase/bbui"
|
||||||
import Dropzone from "components/common/Dropzone.svelte"
|
import Dropzone from "components/common/Dropzone.svelte"
|
||||||
import { capitalise } from "../../../helpers"
|
import { capitalise } from "../../../helpers"
|
||||||
import LinkedRecordSelector from "components/common/LinkedRecordSelector.svelte"
|
import LinkedRowSelector from "components/common/LinkedRowSelector.svelte"
|
||||||
|
|
||||||
export let meta
|
export let meta
|
||||||
export let value = meta.type === "boolean" ? false : ""
|
export let value = meta.type === "boolean" ? false : ""
|
||||||
|
@ -28,7 +28,7 @@
|
||||||
{:else if type === 'boolean'}
|
{:else if type === 'boolean'}
|
||||||
<Toggle text={label} bind:checked={value} data-cy="{meta.name}-input" />
|
<Toggle text={label} bind:checked={value} data-cy="{meta.name}-input" />
|
||||||
{:else if type === 'link'}
|
{:else if type === 'link'}
|
||||||
<LinkedRecordSelector bind:linkedRecords={value} schema={meta} />
|
<LinkedRowSelector bind:linkedRows={value} schema={meta} />
|
||||||
{:else}
|
{:else}
|
||||||
<Input thin {label} data-cy="{meta.name}-input" {type} bind:value />
|
<Input thin {label} data-cy="{meta.name}-input" {type} bind:value />
|
||||||
{/if}
|
{/if}
|
|
@ -10,7 +10,7 @@
|
||||||
import ActionButton from "components/common/ActionButton.svelte"
|
import ActionButton from "components/common/ActionButton.svelte"
|
||||||
import AttachmentList from "./AttachmentList.svelte"
|
import AttachmentList from "./AttachmentList.svelte"
|
||||||
import TablePagination from "./TablePagination.svelte"
|
import TablePagination from "./TablePagination.svelte"
|
||||||
import CreateEditRecordModal from "./modals/CreateEditRecordModal.svelte"
|
import CreateEditRowModal from "./modals/CreateEditRowModal.svelte"
|
||||||
import RowPopover from "./buttons/CreateRowButton.svelte"
|
import RowPopover from "./buttons/CreateRowButton.svelte"
|
||||||
import ColumnPopover from "./buttons/CreateColumnButton.svelte"
|
import ColumnPopover from "./buttons/CreateColumnButton.svelte"
|
||||||
import ViewPopover from "./buttons/CreateViewButton.svelte"
|
import ViewPopover from "./buttons/CreateViewButton.svelte"
|
||||||
|
@ -39,14 +39,14 @@
|
||||||
currentPage * ITEMS_PER_PAGE + ITEMS_PER_PAGE
|
currentPage * ITEMS_PER_PAGE + ITEMS_PER_PAGE
|
||||||
)
|
)
|
||||||
: []
|
: []
|
||||||
$: modelId = data?.length ? data[0].modelId : null
|
$: tableId = data?.length ? data[0].tableId : null
|
||||||
|
|
||||||
function selectRelationship(record, fieldName) {
|
function selectRelationship(row, fieldName) {
|
||||||
if (!record?.[fieldName]?.length) {
|
if (!row?.[fieldName]?.length) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
$goto(
|
$goto(
|
||||||
`/${$params.application}/backend/model/${modelId}/relationship/${record._id}/${fieldName}`
|
`/${$params.application}/backend/table/${tableId}/relationship/${row._id}/${fieldName}`
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -12,7 +12,7 @@
|
||||||
|
|
||||||
$: name = view.name
|
$: name = view.name
|
||||||
|
|
||||||
// Fetch records for specified view
|
// Fetch rows for specified view
|
||||||
$: {
|
$: {
|
||||||
if (!name.startsWith("all_")) {
|
if (!name.startsWith("all_")) {
|
||||||
fetchViewData(name, view.field, view.groupBy)
|
fetchViewData(name, view.field, view.groupBy)
|
||||||
|
|
|
@ -6,22 +6,22 @@ export async function createUser(user) {
|
||||||
return await response.json()
|
return await response.json()
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function saveRecord(record, modelId) {
|
export async function saveRow(row, tableId) {
|
||||||
const SAVE_RECORDS_URL = `/api/${modelId}/records`
|
const SAVE_ROWS_URL = `/api/${tableId}/rows`
|
||||||
const response = await api.post(SAVE_RECORDS_URL, record)
|
const response = await api.post(SAVE_ROWS_URL, row)
|
||||||
|
|
||||||
return await response.json()
|
return await response.json()
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleteRecord(record) {
|
export async function deleteRow(row) {
|
||||||
const DELETE_RECORDS_URL = `/api/${record.modelId}/records/${record._id}/${record._rev}`
|
const DELETE_ROWS_URL = `/api/${row.tableId}/rows/${row._id}/${row._rev}`
|
||||||
const response = await api.delete(DELETE_RECORDS_URL)
|
const response = await api.delete(DELETE_ROWS_URL)
|
||||||
return response
|
return response
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function fetchDataForView(view) {
|
export async function fetchDataForView(view) {
|
||||||
const FETCH_RECORDS_URL = `/api/views/${view.name}`
|
const FETCH_ROWS_URL = `/api/views/${view.name}`
|
||||||
|
|
||||||
const response = await api.get(FETCH_RECORDS_URL)
|
const response = await api.get(FETCH_ROWS_URL)
|
||||||
return await response.json()
|
return await response.json()
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<script>
|
<script>
|
||||||
import { TextButton as Button, Icon, Modal } from "@budibase/bbui"
|
import { TextButton as Button, Icon, Modal } from "@budibase/bbui"
|
||||||
import CreateEditRecordModal from "../modals/CreateEditRecordModal.svelte"
|
import CreateEditRowModal from "../modals/CreateEditRowModal.svelte"
|
||||||
|
|
||||||
let modal
|
let modal
|
||||||
</script>
|
</script>
|
||||||
|
@ -12,5 +12,5 @@
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
<Modal bind:this={modal}>
|
<Modal bind:this={modal}>
|
||||||
<CreateEditRecordModal />
|
<CreateEditRowModal />
|
||||||
</Modal>
|
</Modal>
|
||||||
|
|
|
@ -1,46 +0,0 @@
|
||||||
<script>
|
|
||||||
import { backendUiStore } from "builderStore"
|
|
||||||
import { notifier } from "builderStore/store/notifications"
|
|
||||||
import RecordFieldControl from "../RecordFieldControl.svelte"
|
|
||||||
import * as api from "../api"
|
|
||||||
import { ModalContent } from "@budibase/bbui"
|
|
||||||
import ErrorsBox from "components/common/ErrorsBox.svelte"
|
|
||||||
|
|
||||||
export let record = {}
|
|
||||||
|
|
||||||
let errors = []
|
|
||||||
|
|
||||||
$: creating = record?._id == null
|
|
||||||
$: model = record.modelId
|
|
||||||
? $backendUiStore.models.find(model => model._id === record?.modelId)
|
|
||||||
: $backendUiStore.selectedModel
|
|
||||||
$: modelSchema = Object.entries(model?.schema ?? {})
|
|
||||||
|
|
||||||
async function saveRecord() {
|
|
||||||
const recordResponse = await api.saveRecord(
|
|
||||||
{ ...record, modelId: model._id },
|
|
||||||
model._id
|
|
||||||
)
|
|
||||||
if (recordResponse.errors) {
|
|
||||||
errors = Object.keys(recordResponse.errors)
|
|
||||||
.map(k => ({ dataPath: k, message: recordResponse.errors[k] }))
|
|
||||||
.flat()
|
|
||||||
// Prevent modal closing if there were errors
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
notifier.success("Record saved successfully.")
|
|
||||||
backendUiStore.actions.records.save(recordResponse)
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<ModalContent
|
|
||||||
title={creating ? 'Create Row' : 'Edit Row'}
|
|
||||||
confirmText={creating ? 'Create Row' : 'Save Row'}
|
|
||||||
onConfirm={saveRecord}>
|
|
||||||
<ErrorsBox {errors} />
|
|
||||||
{#each modelSchema as [key, meta]}
|
|
||||||
<div>
|
|
||||||
<RecordFieldControl {meta} bind:value={record[key]} />
|
|
||||||
</div>
|
|
||||||
{/each}
|
|
||||||
</ModalContent>
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
<script>
|
||||||
|
import { backendUiStore } from "builderStore"
|
||||||
|
import { notifier } from "builderStore/store/notifications"
|
||||||
|
import RowFieldControl from "../RowFieldControl.svelte"
|
||||||
|
import * as api from "../api"
|
||||||
|
import { ModalContent } from "@budibase/bbui"
|
||||||
|
import ErrorsBox from "components/common/ErrorsBox.svelte"
|
||||||
|
|
||||||
|
export let row = {}
|
||||||
|
|
||||||
|
let errors = []
|
||||||
|
|
||||||
|
$: creating = row?._id == null
|
||||||
|
$: table = row.tableId
|
||||||
|
? $backendUiStore.tables.find(table => table._id === row?.tableId)
|
||||||
|
: $backendUiStore.selectedTable
|
||||||
|
$: tableSchema = Object.entries(table?.schema ?? {})
|
||||||
|
|
||||||
|
async function saveRow() {
|
||||||
|
const rowResponse = await api.saveRow(
|
||||||
|
{ ...row, tableId: table._id },
|
||||||
|
table._id
|
||||||
|
)
|
||||||
|
if (rowResponse.errors) {
|
||||||
|
errors = Object.keys(rowResponse.errors)
|
||||||
|
.map(k => ({ dataPath: k, message: rowResponse.errors[k] }))
|
||||||
|
.flat()
|
||||||
|
// Prevent modal closing if there were errors
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
notifier.success("Row saved successfully.")
|
||||||
|
backendUiStore.actions.rows.save(rowResponse)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<ModalContent
|
||||||
|
title={creating ? 'Create Row' : 'Edit Row'}
|
||||||
|
confirmText={creating ? 'Create Row' : 'Save Row'}
|
||||||
|
onConfirm={saveRow}>
|
||||||
|
<ErrorsBox {errors} />
|
||||||
|
{#each tableSchema as [key, meta]}
|
||||||
|
<div>
|
||||||
|
<RowFieldControl {meta} bind:value={row[key]} />
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</ModalContent>
|
|
@ -14,13 +14,13 @@
|
||||||
export let view = {}
|
export let view = {}
|
||||||
export let onClosed
|
export let onClosed
|
||||||
|
|
||||||
$: viewModel = $backendUiStore.models.find(
|
$: viewTable = $backendUiStore.tables.find(
|
||||||
({ _id }) => _id === $backendUiStore.selectedView.modelId
|
({ _id }) => _id === $backendUiStore.selectedView.tableId
|
||||||
)
|
)
|
||||||
$: fields =
|
$: fields =
|
||||||
viewModel &&
|
viewTable &&
|
||||||
Object.keys(viewModel.schema).filter(
|
Object.keys(viewTable.schema).filter(
|
||||||
field => viewModel.schema[field].type === "number"
|
field => viewTable.schema[field].type === "number"
|
||||||
)
|
)
|
||||||
|
|
||||||
function saveView() {
|
function saveView() {
|
||||||
|
|
|
@ -32,10 +32,10 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
function deleteColumn() {
|
function deleteColumn() {
|
||||||
if (field.name === $backendUiStore.selectedModel.primaryDisplay) {
|
if (field.name === $backendUiStore.selectedTable.primaryDisplay) {
|
||||||
notifier.danger("You cannot delete the primary display column")
|
notifier.danger("You cannot delete the primary display column")
|
||||||
} else {
|
} else {
|
||||||
backendUiStore.actions.models.deleteField(field)
|
backendUiStore.actions.tables.deleteField(field)
|
||||||
notifier.success("Column deleted")
|
notifier.success("Column deleted")
|
||||||
}
|
}
|
||||||
hideEditor()
|
hideEditor()
|
||||||
|
|
|
@ -19,7 +19,7 @@
|
||||||
import Checkbox from "components/common/Checkbox.svelte"
|
import Checkbox from "components/common/Checkbox.svelte"
|
||||||
import ActionButton from "components/common/ActionButton.svelte"
|
import ActionButton from "components/common/ActionButton.svelte"
|
||||||
import DatePicker from "components/common/DatePicker.svelte"
|
import DatePicker from "components/common/DatePicker.svelte"
|
||||||
import LinkedRecordSelector from "components/common/LinkedRecordSelector.svelte"
|
import LinkedRowSelector from "components/common/LinkedRowSelector.svelte"
|
||||||
import * as api from "../api"
|
import * as api from "../api"
|
||||||
|
|
||||||
let fieldDefinitions = cloneDeep(FIELDS)
|
let fieldDefinitions = cloneDeep(FIELDS)
|
||||||
|
@ -31,14 +31,14 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
let originalName = field.name
|
let originalName = field.name
|
||||||
$: modelOptions = $backendUiStore.models.filter(
|
$: tableOptions = $backendUiStore.tables.filter(
|
||||||
model => model._id !== $backendUiStore.draftModel._id
|
table => table._id !== $backendUiStore.draftTable._id
|
||||||
)
|
)
|
||||||
$: required = !!field?.constraints?.presence
|
$: required = !!field?.constraints?.presence
|
||||||
|
|
||||||
async function saveColumn() {
|
async function saveColumn() {
|
||||||
backendUiStore.update(state => {
|
backendUiStore.update(state => {
|
||||||
backendUiStore.actions.models.saveField({
|
backendUiStore.actions.tables.saveField({
|
||||||
originalName,
|
originalName,
|
||||||
field,
|
field,
|
||||||
})
|
})
|
||||||
|
@ -111,10 +111,10 @@
|
||||||
label="Max Value"
|
label="Max Value"
|
||||||
bind:value={field.constraints.numericality.lessThanOrEqualTo} />
|
bind:value={field.constraints.numericality.lessThanOrEqualTo} />
|
||||||
{:else if field.type === 'link'}
|
{:else if field.type === 'link'}
|
||||||
<Select label="Table" thin secondary bind:value={field.modelId}>
|
<Select label="Table" thin secondary bind:value={field.tableId}>
|
||||||
<option value="">Choose an option</option>
|
<option value="">Choose an option</option>
|
||||||
{#each modelOptions as model}
|
{#each tableOptions as table}
|
||||||
<option value={model._id}>{model.name}</option>
|
<option value={table._id}>{table.name}</option>
|
||||||
{/each}
|
{/each}
|
||||||
</Select>
|
</Select>
|
||||||
<Input
|
<Input
|
||||||
|
|
|
@ -10,11 +10,11 @@
|
||||||
let name
|
let name
|
||||||
let field
|
let field
|
||||||
|
|
||||||
$: fields = Object.keys($backendUiStore.selectedModel.schema).filter(key => {
|
$: fields = Object.keys($backendUiStore.selectedTable.schema).filter(key => {
|
||||||
return $backendUiStore.selectedModel.schema[key].type === "number"
|
return $backendUiStore.selectedTable.schema[key].type === "number"
|
||||||
})
|
})
|
||||||
$: views = $backendUiStore.models.flatMap(model =>
|
$: views = $backendUiStore.tables.flatMap(table =>
|
||||||
Object.keys(model.views || {})
|
Object.keys(table.views || {})
|
||||||
)
|
)
|
||||||
|
|
||||||
function saveView() {
|
function saveView() {
|
||||||
|
@ -24,7 +24,7 @@
|
||||||
}
|
}
|
||||||
backendUiStore.actions.views.save({
|
backendUiStore.actions.views.save({
|
||||||
name,
|
name,
|
||||||
modelId: $backendUiStore.selectedModel._id,
|
tableId: $backendUiStore.selectedTable._id,
|
||||||
field,
|
field,
|
||||||
})
|
})
|
||||||
notifier.success(`View ${name} created`)
|
notifier.success(`View ${name} created`)
|
||||||
|
|
|
@ -45,10 +45,10 @@
|
||||||
export let view = {}
|
export let view = {}
|
||||||
export let onClosed
|
export let onClosed
|
||||||
|
|
||||||
$: viewModel = $backendUiStore.models.find(
|
$: viewTable = $backendUiStore.tables.find(
|
||||||
({ _id }) => _id === $backendUiStore.selectedView.modelId
|
({ _id }) => _id === $backendUiStore.selectedView.tableId
|
||||||
)
|
)
|
||||||
$: fields = viewModel && Object.keys(viewModel.schema)
|
$: fields = viewTable && Object.keys(viewTable.schema)
|
||||||
|
|
||||||
function saveView() {
|
function saveView() {
|
||||||
backendUiStore.actions.views.save(view)
|
backendUiStore.actions.views.save(view)
|
||||||
|
@ -71,25 +71,25 @@
|
||||||
|
|
||||||
function isMultipleChoice(field) {
|
function isMultipleChoice(field) {
|
||||||
return (
|
return (
|
||||||
(viewModel.schema[field].constraints &&
|
(viewTable.schema[field].constraints &&
|
||||||
viewModel.schema[field].constraints.inclusion &&
|
viewTable.schema[field].constraints.inclusion &&
|
||||||
viewModel.schema[field].constraints.inclusion.length) ||
|
viewTable.schema[field].constraints.inclusion.length) ||
|
||||||
viewModel.schema[field].type === "boolean"
|
viewTable.schema[field].type === "boolean"
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
function fieldOptions(field) {
|
function fieldOptions(field) {
|
||||||
return viewModel.schema[field].type === "options"
|
return viewTable.schema[field].type === "options"
|
||||||
? viewModel.schema[field].constraints.inclusion
|
? viewTable.schema[field].constraints.inclusion
|
||||||
: [true, false]
|
: [true, false]
|
||||||
}
|
}
|
||||||
|
|
||||||
function isDate(field) {
|
function isDate(field) {
|
||||||
return viewModel.schema[field].type === "datetime"
|
return viewTable.schema[field].type === "datetime"
|
||||||
}
|
}
|
||||||
|
|
||||||
function isNumber(field) {
|
function isNumber(field) {
|
||||||
return viewModel.schema[field].type === "number"
|
return viewTable.schema[field].type === "number"
|
||||||
}
|
}
|
||||||
|
|
||||||
const fieldChanged = filter => ev => {
|
const fieldChanged = filter => ev => {
|
||||||
|
@ -97,8 +97,8 @@
|
||||||
if (
|
if (
|
||||||
filter.key &&
|
filter.key &&
|
||||||
ev.target.value &&
|
ev.target.value &&
|
||||||
viewModel.schema[filter.key].type !==
|
viewTable.schema[filter.key].type !==
|
||||||
viewModel.schema[ev.target.value].type
|
viewTable.schema[ev.target.value].type
|
||||||
) {
|
) {
|
||||||
filter.value = ""
|
filter.value = ""
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,10 +6,10 @@
|
||||||
export let view = {}
|
export let view = {}
|
||||||
export let onClosed
|
export let onClosed
|
||||||
|
|
||||||
$: viewModel = $backendUiStore.models.find(
|
$: viewTable = $backendUiStore.tables.find(
|
||||||
({ _id }) => _id === $backendUiStore.selectedView.modelId
|
({ _id }) => _id === $backendUiStore.selectedView.tableId
|
||||||
)
|
)
|
||||||
$: fields = viewModel && Object.keys(viewModel.schema)
|
$: fields = viewTable && Object.keys(viewTable.schema)
|
||||||
|
|
||||||
function saveView() {
|
function saveView() {
|
||||||
backendUiStore.actions.views.save(view)
|
backendUiStore.actions.views.save(view)
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
<script>
|
<script>
|
||||||
import { backendUiStore } from "builderStore"
|
import { backendUiStore } from "builderStore"
|
||||||
import { DropdownMenu, Icon, Modal } from "@budibase/bbui"
|
import { DropdownMenu, Icon, Modal } from "@budibase/bbui"
|
||||||
import CreateEditRecordModal from "../modals/CreateEditRecordModal.svelte"
|
import CreateEditRowModal from "../modals/CreateEditRowModal.svelte"
|
||||||
import * as api from "../api"
|
import * as api from "../api"
|
||||||
import { notifier } from "builderStore/store/notifications"
|
import { notifier } from "builderStore/store/notifications"
|
||||||
import ConfirmDialog from "components/common/ConfirmDialog.svelte"
|
import ConfirmDialog from "components/common/ConfirmDialog.svelte"
|
||||||
|
@ -24,9 +24,9 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
async function deleteRow() {
|
async function deleteRow() {
|
||||||
await api.deleteRecord(row)
|
await api.deleteRow(row)
|
||||||
notifier.success("Record deleted")
|
notifier.success("Row deleted")
|
||||||
backendUiStore.actions.records.delete(row)
|
backendUiStore.actions.rows.delete(row)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -52,7 +52,7 @@
|
||||||
onOk={deleteRow}
|
onOk={deleteRow}
|
||||||
title="Confirm Delete" />
|
title="Confirm Delete" />
|
||||||
<Modal bind:this={modal}>
|
<Modal bind:this={modal}>
|
||||||
<CreateEditRecordModal record={row} />
|
<CreateEditRowModal {row} />
|
||||||
</Modal>
|
</Modal>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
data-cy="model-nav-item"
|
data-cy="table-nav-item"
|
||||||
class:indented
|
class:indented
|
||||||
class:selected
|
class:selected
|
||||||
on:click
|
on:click
|
|
@ -21,28 +21,28 @@
|
||||||
!schema || Object.keys(schema).every(column => schema[column].success)
|
!schema || Object.keys(schema).every(column => schema[column].success)
|
||||||
$: dataImport = {
|
$: dataImport = {
|
||||||
valid,
|
valid,
|
||||||
schema: buildModelSchema(schema),
|
schema: buildTableSchema(schema),
|
||||||
path: files[0] && files[0].path,
|
path: files[0] && files[0].path,
|
||||||
}
|
}
|
||||||
|
|
||||||
function buildModelSchema(schema) {
|
function buildTableSchema(schema) {
|
||||||
const modelSchema = {}
|
const tableSchema = {}
|
||||||
for (let key in schema) {
|
for (let key in schema) {
|
||||||
const type = schema[key].type
|
const type = schema[key].type
|
||||||
|
|
||||||
if (type === "omit") continue
|
if (type === "omit") continue
|
||||||
|
|
||||||
modelSchema[key] = {
|
tableSchema[key] = {
|
||||||
name: key,
|
name: key,
|
||||||
type,
|
type,
|
||||||
constraints: FIELDS[type.toUpperCase()].constraints,
|
constraints: FIELDS[type.toUpperCase()].constraints,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return modelSchema
|
return tableSchema
|
||||||
}
|
}
|
||||||
|
|
||||||
async function validateCSV() {
|
async function validateCSV() {
|
||||||
const response = await api.post("/api/models/csv/validate", {
|
const response = await api.post("/api/tables/csv/validate", {
|
||||||
file: files[0],
|
file: files[0],
|
||||||
schema: schema || {},
|
schema: schema || {},
|
||||||
})
|
})
|
|
@ -11,9 +11,9 @@
|
||||||
$: selectedView =
|
$: selectedView =
|
||||||
$backendUiStore.selectedView && $backendUiStore.selectedView.name
|
$backendUiStore.selectedView && $backendUiStore.selectedView.name
|
||||||
|
|
||||||
function selectModel(model) {
|
function selectTable(table) {
|
||||||
backendUiStore.actions.models.select(model)
|
backendUiStore.actions.tables.select(table)
|
||||||
$goto(`./model/${model._id}`)
|
$goto(`./table/${table._id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
function selectView(view) {
|
function selectView(view) {
|
||||||
|
@ -30,15 +30,15 @@
|
||||||
<Spacer medium />
|
<Spacer medium />
|
||||||
<CreateTableModal />
|
<CreateTableModal />
|
||||||
<div class="hierarchy-items-container">
|
<div class="hierarchy-items-container">
|
||||||
{#each $backendUiStore.models as model}
|
{#each $backendUiStore.tables as table}
|
||||||
<ListItem
|
<ListItem
|
||||||
selected={selectedView === `all_${model._id}`}
|
selected={selectedView === `all_${table._id}`}
|
||||||
title={model.name}
|
title={table.name}
|
||||||
icon="ri-table-fill"
|
icon="ri-table-fill"
|
||||||
on:click={() => selectModel(model)}>
|
on:click={() => selectTable(table)}>
|
||||||
<EditTablePopover table={model} />
|
<EditTablePopover {table} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
{#each Object.keys(model.views || {}) as viewName}
|
{#each Object.keys(table.views || {}) as viewName}
|
||||||
<ListItem
|
<ListItem
|
||||||
indented
|
indented
|
||||||
selected={selectedView === viewName}
|
selected={selectedView === viewName}
|
||||||
|
@ -46,10 +46,10 @@
|
||||||
icon="ri-eye-line"
|
icon="ri-eye-line"
|
||||||
on:click={() => (selectedView === viewName ? {} : selectView({
|
on:click={() => (selectedView === viewName ? {} : selectView({
|
||||||
name: viewName,
|
name: viewName,
|
||||||
...model.views[viewName],
|
...table.views[viewName],
|
||||||
}))}>
|
}))}>
|
||||||
<EditViewPopover
|
<EditViewPopover
|
||||||
view={{ name: viewName, ...model.views[viewName] }} />
|
view={{ name: viewName, ...table.views[viewName] }} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
{/each}
|
{/each}
|
||||||
{/each}
|
{/each}
|
|
@ -7,14 +7,14 @@
|
||||||
import TableDataImport from "../TableDataImport.svelte"
|
import TableDataImport from "../TableDataImport.svelte"
|
||||||
import analytics from "analytics"
|
import analytics from "analytics"
|
||||||
import screenTemplates from "builderStore/store/screenTemplates"
|
import screenTemplates from "builderStore/store/screenTemplates"
|
||||||
import { NEW_RECORD_TEMPLATE } from "builderStore/store/screenTemplates/newRecordScreen"
|
import { NEW_ROW_TEMPLATE } from "builderStore/store/screenTemplates/newRowScreen"
|
||||||
import { RECORD_DETAIL_TEMPLATE } from "builderStore/store/screenTemplates/recordDetailScreen"
|
import { ROW_DETAIL_TEMPLATE } from "builderStore/store/screenTemplates/rowDetailScreen"
|
||||||
import { RECORD_LIST_TEMPLATE } from "builderStore/store/screenTemplates/recordListScreen"
|
import { ROW_LIST_TEMPLATE } from "builderStore/store/screenTemplates/rowListScreen"
|
||||||
|
|
||||||
const defaultScreens = [
|
const defaultScreens = [
|
||||||
NEW_RECORD_TEMPLATE,
|
NEW_ROW_TEMPLATE,
|
||||||
RECORD_DETAIL_TEMPLATE,
|
ROW_DETAIL_TEMPLATE,
|
||||||
RECORD_LIST_TEMPLATE,
|
ROW_LIST_TEMPLATE,
|
||||||
]
|
]
|
||||||
|
|
||||||
let modal
|
let modal
|
||||||
|
@ -27,16 +27,16 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
async function saveTable() {
|
async function saveTable() {
|
||||||
const model = await backendUiStore.actions.models.save({
|
const table = await backendUiStore.actions.tables.save({
|
||||||
name,
|
name,
|
||||||
schema: dataImport.schema || {},
|
schema: dataImport.schema || {},
|
||||||
dataImport,
|
dataImport,
|
||||||
})
|
})
|
||||||
notifier.success(`Table ${name} created successfully.`)
|
notifier.success(`Table ${name} created successfully.`)
|
||||||
$goto(`./model/${model._id}`)
|
$goto(`./table/${table._id}`)
|
||||||
analytics.captureEvent("Table Created", { name })
|
analytics.captureEvent("Table Created", { name })
|
||||||
|
|
||||||
const screens = screenTemplates($store, [model])
|
const screens = screenTemplates($store, [table])
|
||||||
.filter(template => defaultScreens.includes(template.id))
|
.filter(template => defaultScreens.includes(template.id))
|
||||||
.map(template => template.create())
|
.map(template => template.create())
|
||||||
|
|
||||||
|
@ -46,9 +46,9 @@
|
||||||
} catch (_) {
|
} catch (_) {
|
||||||
// TODO: this is temporary
|
// TODO: this is temporary
|
||||||
// a cypress test is failing, because I added the
|
// a cypress test is failing, because I added the
|
||||||
// NewRecord component. So - this throws an exception
|
// NewRow component. So - this throws an exception
|
||||||
// because the currently released standard-components (on NPM)
|
// because the currently released standard-components (on NPM)
|
||||||
// does not have NewRecord
|
// does not have NewRow
|
||||||
// we should remove this after this has been released
|
// we should remove this after this has been released
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -29,13 +29,13 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
async function deleteTable() {
|
async function deleteTable() {
|
||||||
await backendUiStore.actions.models.delete(table)
|
await backendUiStore.actions.tables.delete(table)
|
||||||
notifier.success("Table deleted")
|
notifier.success("Table deleted")
|
||||||
hideEditor()
|
hideEditor()
|
||||||
}
|
}
|
||||||
|
|
||||||
async function save() {
|
async function save() {
|
||||||
await backendUiStore.actions.models.save(table)
|
await backendUiStore.actions.tables.save(table)
|
||||||
notifier.success("Table renamed successfully")
|
notifier.success("Table renamed successfully")
|
||||||
hideEditor()
|
hideEditor()
|
||||||
}
|
}
|
|
@ -39,10 +39,10 @@
|
||||||
|
|
||||||
async function deleteView() {
|
async function deleteView() {
|
||||||
const name = view.name
|
const name = view.name
|
||||||
const id = view.modelId
|
const id = view.tableId
|
||||||
await backendUiStore.actions.views.delete(name)
|
await backendUiStore.actions.views.delete(name)
|
||||||
notifier.success("View deleted")
|
notifier.success("View deleted")
|
||||||
$goto(`./model/${id}`)
|
$goto(`./table/${id}`)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -1,53 +0,0 @@
|
||||||
<script>
|
|
||||||
import { onMount } from "svelte"
|
|
||||||
import { backendUiStore } from "builderStore"
|
|
||||||
import api from "builderStore/api"
|
|
||||||
import { Select, Label, Multiselect } from "@budibase/bbui"
|
|
||||||
import { capitalise } from "../../helpers"
|
|
||||||
|
|
||||||
export let schema
|
|
||||||
export let linkedRecords = []
|
|
||||||
|
|
||||||
let records = []
|
|
||||||
|
|
||||||
$: label = capitalise(schema.name)
|
|
||||||
$: linkedModelId = schema.modelId
|
|
||||||
$: linkedModel = $backendUiStore.models.find(
|
|
||||||
model => model._id === linkedModelId
|
|
||||||
)
|
|
||||||
$: fetchRecords(linkedModelId)
|
|
||||||
|
|
||||||
async function fetchRecords(linkedModelId) {
|
|
||||||
const FETCH_RECORDS_URL = `/api/${linkedModelId}/records`
|
|
||||||
try {
|
|
||||||
const response = await api.get(FETCH_RECORDS_URL)
|
|
||||||
records = await response.json()
|
|
||||||
} catch (error) {
|
|
||||||
console.log(error)
|
|
||||||
records = []
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function getPrettyName(record) {
|
|
||||||
return record[linkedModel.primaryDisplay || "_id"]
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
{#if linkedModel.primaryDisplay == null}
|
|
||||||
<Label extraSmall grey>{label}</Label>
|
|
||||||
<Label small black>
|
|
||||||
Please choose a primary display column for the
|
|
||||||
<b>{linkedModel.name}</b>
|
|
||||||
table.
|
|
||||||
</Label>
|
|
||||||
{:else}
|
|
||||||
<Multiselect
|
|
||||||
secondary
|
|
||||||
bind:value={linkedRecords}
|
|
||||||
{label}
|
|
||||||
placeholder="Choose some options">
|
|
||||||
{#each records as record}
|
|
||||||
<option value={record._id}>{getPrettyName(record)}</option>
|
|
||||||
{/each}
|
|
||||||
</Multiselect>
|
|
||||||
{/if}
|
|
|
@ -0,0 +1,53 @@
|
||||||
|
<script>
|
||||||
|
import { onMount } from "svelte"
|
||||||
|
import { backendUiStore } from "builderStore"
|
||||||
|
import api from "builderStore/api"
|
||||||
|
import { Select, Label, Multiselect } from "@budibase/bbui"
|
||||||
|
import { capitalise } from "../../helpers"
|
||||||
|
|
||||||
|
export let schema
|
||||||
|
export let linkedRows = []
|
||||||
|
|
||||||
|
let rows = []
|
||||||
|
|
||||||
|
$: label = capitalise(schema.name)
|
||||||
|
$: linkedTableId = schema.tableId
|
||||||
|
$: linkedTable = $backendUiStore.tables.find(
|
||||||
|
table => table._id === linkedTableId
|
||||||
|
)
|
||||||
|
$: fetchRows(linkedTableId)
|
||||||
|
|
||||||
|
async function fetchRows(linkedTableId) {
|
||||||
|
const FETCH_ROWS_URL = `/api/${linkedTableId}/rows`
|
||||||
|
try {
|
||||||
|
const response = await api.get(FETCH_ROWS_URL)
|
||||||
|
rows = await response.json()
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error)
|
||||||
|
rows = []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPrettyName(row) {
|
||||||
|
return row[linkedTable.primaryDisplay || "_id"]
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{#if linkedTable.primaryDisplay == null}
|
||||||
|
<Label extraSmall grey>{label}</Label>
|
||||||
|
<Label small black>
|
||||||
|
Please choose a primary display column for the
|
||||||
|
<b>{linkedTable.name}</b>
|
||||||
|
table.
|
||||||
|
</Label>
|
||||||
|
{:else}
|
||||||
|
<Multiselect
|
||||||
|
secondary
|
||||||
|
bind:value={linkedRows}
|
||||||
|
{label}
|
||||||
|
placeholder="Choose some options">
|
||||||
|
{#each rows as row}
|
||||||
|
<option value={row._id}>{getPrettyName(row)}</option>
|
||||||
|
{/each}
|
||||||
|
</Multiselect>
|
||||||
|
{/if}
|
|
@ -10,23 +10,23 @@
|
||||||
componentInstanceId: $store.currentComponentInfo._id,
|
componentInstanceId: $store.currentComponentInfo._id,
|
||||||
components: $store.components,
|
components: $store.components,
|
||||||
screen: $store.currentPreviewItem,
|
screen: $store.currentPreviewItem,
|
||||||
models: $backendUiStore.models,
|
tables: $backendUiStore.tables,
|
||||||
})
|
})
|
||||||
|
|
||||||
// just wraps binding in {{ ... }}
|
// just wraps binding in {{ ... }}
|
||||||
const toBindingExpression = bindingPath => `{{ ${bindingPath} }}`
|
const toBindingExpression = bindingPath => `{{ ${bindingPath} }}`
|
||||||
|
|
||||||
const modelFields = modelId => {
|
const tableFields = tableId => {
|
||||||
const model = $backendUiStore.models.find(m => m._id === modelId)
|
const table = $backendUiStore.tables.find(m => m._id === tableId)
|
||||||
|
|
||||||
return Object.keys(model.schema).map(k => ({
|
return Object.keys(table.schema).map(k => ({
|
||||||
name: k,
|
name: k,
|
||||||
type: model.schema[k].type,
|
type: table.schema[k].type,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
$: schemaFields =
|
$: schemaFields =
|
||||||
parameters && parameters.modelId ? modelFields(parameters.modelId) : []
|
parameters && parameters.tableId ? tableFields(parameters.tableId) : []
|
||||||
|
|
||||||
const onFieldsChanged = e => {
|
const onFieldsChanged = e => {
|
||||||
parameters.fields = e.detail
|
parameters.fields = e.detail
|
||||||
|
@ -35,14 +35,14 @@
|
||||||
|
|
||||||
<div class="root">
|
<div class="root">
|
||||||
<Label size="m" color="dark">Table</Label>
|
<Label size="m" color="dark">Table</Label>
|
||||||
<Select secondary bind:value={parameters.modelId}>
|
<Select secondary bind:value={parameters.tableId}>
|
||||||
<option value="" />
|
<option value="" />
|
||||||
{#each $backendUiStore.models as model}
|
{#each $backendUiStore.tables as table}
|
||||||
<option value={model._id}>{model.name}</option>
|
<option value={table._id}>{table.name}</option>
|
||||||
{/each}
|
{/each}
|
||||||
</Select>
|
</Select>
|
||||||
|
|
||||||
{#if parameters.modelId}
|
{#if parameters.tableId}
|
||||||
<SaveFields
|
<SaveFields
|
||||||
parameterFields={parameters.fields}
|
parameterFields={parameters.fields}
|
||||||
{schemaFields}
|
{schemaFields}
|
|
@ -35,7 +35,7 @@
|
||||||
componentInstanceId: $store.currentComponentInfo._id,
|
componentInstanceId: $store.currentComponentInfo._id,
|
||||||
components: $store.components,
|
components: $store.components,
|
||||||
screen: $store.currentPreviewItem,
|
screen: $store.currentPreviewItem,
|
||||||
models: $backendUiStore.models,
|
tables: $backendUiStore.tables,
|
||||||
})
|
})
|
||||||
|
|
||||||
const addField = () => {
|
const addField = () => {
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
runtimeToReadableBinding,
|
runtimeToReadableBinding,
|
||||||
} from "builderStore/replaceBindings"
|
} from "builderStore/replaceBindings"
|
||||||
|
|
||||||
// parameters.contextPath used in the client handler to determine which record to save
|
// parameters.contextPath used in the client handler to determine which row to save
|
||||||
// this could be "data" or "data.parent", "data.parent.parent" etc
|
// this could be "data" or "data.parent", "data.parent.parent" etc
|
||||||
export let parameters
|
export let parameters
|
||||||
|
|
||||||
|
@ -19,7 +19,7 @@
|
||||||
componentInstanceId: $store.currentComponentInfo._id,
|
componentInstanceId: $store.currentComponentInfo._id,
|
||||||
components: $store.components,
|
components: $store.components,
|
||||||
screen: $store.currentPreviewItem,
|
screen: $store.currentPreviewItem,
|
||||||
models: $backendUiStore.models,
|
tables: $backendUiStore.tables,
|
||||||
})
|
})
|
||||||
|
|
||||||
$: {
|
$: {
|
||||||
|
@ -65,18 +65,18 @@
|
||||||
|
|
||||||
const component = $store.components[instance._component]
|
const component = $store.components[instance._component]
|
||||||
|
|
||||||
// component.context is the name of the prop that holds the modelId
|
// component.context is the name of the prop that holds the tableId
|
||||||
const modelInfo = instance[component.context]
|
const tableInfo = instance[component.context]
|
||||||
const modelId =
|
const tableId =
|
||||||
typeof modelInfo === "string" ? modelInfo : modelInfo.modelId
|
typeof tableInfo === "string" ? tableInfo : tableInfo.tableId
|
||||||
|
|
||||||
if (!modelInfo) return []
|
if (!tableInfo) return []
|
||||||
|
|
||||||
const model = $backendUiStore.models.find(m => m._id === modelId)
|
const table = $backendUiStore.tables.find(m => m._id === tableId)
|
||||||
parameters.modelId = modelId
|
parameters.tableId = tableId
|
||||||
return Object.keys(model.schema).map(k => ({
|
return Object.keys(table.schema).map(k => ({
|
||||||
name: k,
|
name: k,
|
||||||
type: model.schema[k].type,
|
type: table.schema[k].type,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,8 +88,8 @@
|
||||||
<div class="root">
|
<div class="root">
|
||||||
{#if idFields.length === 0}
|
{#if idFields.length === 0}
|
||||||
<div class="cannot-use">
|
<div class="cannot-use">
|
||||||
Update record can only be used within a component that provides data, such
|
Update row can only be used within a component that provides data, such as
|
||||||
as a List
|
a List
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else}
|
||||||
<Label size="m" color="dark">Datasource</Label>
|
<Label size="m" color="dark">Datasource</Label>
|
|
@ -14,29 +14,29 @@
|
||||||
componentInstanceId: $store.currentComponentInfo._id,
|
componentInstanceId: $store.currentComponentInfo._id,
|
||||||
components: $store.components,
|
components: $store.components,
|
||||||
screen: $store.currentPreviewItem,
|
screen: $store.currentPreviewItem,
|
||||||
models: $backendUiStore.models,
|
tables: $backendUiStore.tables,
|
||||||
})
|
})
|
||||||
|
|
||||||
let idFields
|
let idFields
|
||||||
let recordId
|
let rowId
|
||||||
$: {
|
$: {
|
||||||
idFields = bindableProperties.filter(
|
idFields = bindableProperties.filter(
|
||||||
bindable =>
|
bindable =>
|
||||||
bindable.type === "context" && bindable.runtimeBinding.endsWith("._id")
|
bindable.type === "context" && bindable.runtimeBinding.endsWith("._id")
|
||||||
)
|
)
|
||||||
// ensure recordId is always defaulted - there is usually only one option
|
// ensure rowId is always defaulted - there is usually only one option
|
||||||
if (idFields.length > 0 && !parameters._id) {
|
if (idFields.length > 0 && !parameters._id) {
|
||||||
recordId = idFields[0].runtimeBinding
|
rowId = idFields[0].runtimeBinding
|
||||||
parameters = parameters
|
parameters = parameters
|
||||||
} else if (!recordId && parameters._id) {
|
} else if (!rowId && parameters._id) {
|
||||||
recordId = parameters._id
|
rowId = parameters._id
|
||||||
.replace("{{", "")
|
.replace("{{", "")
|
||||||
.replace("}}", "")
|
.replace("}}", "")
|
||||||
.trim()
|
.trim()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$: parameters._id = `{{ ${recordId} }}`
|
$: parameters._id = `{{ ${rowId} }}`
|
||||||
|
|
||||||
// just wraps binding in {{ ... }}
|
// just wraps binding in {{ ... }}
|
||||||
const toBindingExpression = bindingPath => `{{ ${bindingPath} }}`
|
const toBindingExpression = bindingPath => `{{ ${bindingPath} }}`
|
||||||
|
@ -44,11 +44,11 @@
|
||||||
// finds the selected idBinding, then reads the table/view
|
// finds the selected idBinding, then reads the table/view
|
||||||
// from the component instance that it belongs to.
|
// from the component instance that it belongs to.
|
||||||
// then returns the field names for that schema
|
// then returns the field names for that schema
|
||||||
const schemaFromIdBinding = recordId => {
|
const schemaFromIdBinding = rowId => {
|
||||||
if (!recordId) return []
|
if (!rowId) return []
|
||||||
|
|
||||||
const idBinding = bindableProperties.find(
|
const idBinding = bindableProperties.find(
|
||||||
prop => prop.runtimeBinding === recordId
|
prop => prop.runtimeBinding === rowId
|
||||||
)
|
)
|
||||||
if (!idBinding) return []
|
if (!idBinding) return []
|
||||||
|
|
||||||
|
@ -56,23 +56,23 @@
|
||||||
|
|
||||||
const component = $store.components[instance._component]
|
const component = $store.components[instance._component]
|
||||||
|
|
||||||
// component.context is the name of the prop that holds the modelId
|
// component.context is the name of the prop that holds the tableId
|
||||||
const modelInfo = instance[component.context]
|
const tableInfo = instance[component.context]
|
||||||
|
|
||||||
if (!modelInfo) return []
|
if (!tableInfo) return []
|
||||||
|
|
||||||
const model = $backendUiStore.models.find(m => m._id === modelInfo.modelId)
|
const table = $backendUiStore.tables.find(m => m._id === tableInfo.tableId)
|
||||||
parameters.modelId = modelInfo.modelId
|
parameters.tableId = tableInfo.tableId
|
||||||
return Object.keys(model.schema).map(k => ({
|
return Object.keys(table.schema).map(k => ({
|
||||||
name: k,
|
name: k,
|
||||||
type: model.schema[k].type,
|
type: table.schema[k].type,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
let schemaFields
|
let schemaFields
|
||||||
$: {
|
$: {
|
||||||
if (parameters && recordId) {
|
if (parameters && rowId) {
|
||||||
schemaFields = schemaFromIdBinding(recordId)
|
schemaFields = schemaFromIdBinding(rowId)
|
||||||
} else {
|
} else {
|
||||||
schemaFields = []
|
schemaFields = []
|
||||||
}
|
}
|
||||||
|
@ -86,12 +86,12 @@
|
||||||
<div class="root">
|
<div class="root">
|
||||||
{#if idFields.length === 0}
|
{#if idFields.length === 0}
|
||||||
<div class="cannot-use">
|
<div class="cannot-use">
|
||||||
Update record can only be used within a component that provides data, such
|
Update row can only be used within a component that provides data, such as
|
||||||
as a List
|
a List
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else}
|
||||||
<Label size="m" color="dark">Record Id</Label>
|
<Label size="m" color="dark">Row Id</Label>
|
||||||
<Select secondary bind:value={recordId}>
|
<Select secondary bind:value={rowId}>
|
||||||
<option value="" />
|
<option value="" />
|
||||||
{#each idFields as idField}
|
{#each idFields as idField}
|
||||||
<option value={idField.runtimeBinding}>
|
<option value={idField.runtimeBinding}>
|
||||||
|
@ -101,7 +101,7 @@
|
||||||
</Select>
|
</Select>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
{#if recordId}
|
{#if rowId}
|
||||||
<SaveFields
|
<SaveFields
|
||||||
parameterFields={parameters.fields}
|
parameterFields={parameters.fields}
|
||||||
{schemaFields}
|
{schemaFields}
|
|
@ -1,5 +1,5 @@
|
||||||
import NavigateTo from "./NavigateTo.svelte"
|
import NavigateTo from "./NavigateTo.svelte"
|
||||||
import SaveRecord from "./SaveRecord.svelte"
|
import SaveRow from "./SaveRow.svelte"
|
||||||
|
|
||||||
// defines what actions are available, when adding a new one
|
// defines what actions are available, when adding a new one
|
||||||
// the component is the setup panel for the action
|
// the component is the setup panel for the action
|
||||||
|
@ -8,8 +8,8 @@ import SaveRecord from "./SaveRecord.svelte"
|
||||||
|
|
||||||
export default [
|
export default [
|
||||||
{
|
{
|
||||||
name: "Save Record",
|
name: "Save Row",
|
||||||
component: SaveRecord,
|
component: SaveRow,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Navigate To",
|
name: "Navigate To",
|
||||||
|
|
|
@ -4568,8 +4568,8 @@ export default [
|
||||||
label: "receipt",
|
label: "receipt",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
value: "fas fa-record-vinyl",
|
value: "fas fa-row-vinyl",
|
||||||
label: "record-vinyl",
|
label: "row-vinyl",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
value: "fas fa-recycle",
|
value: "fas fa-recycle",
|
||||||
|
|
|
@ -14,7 +14,7 @@
|
||||||
let templateIndex
|
let templateIndex
|
||||||
let draftScreen
|
let draftScreen
|
||||||
|
|
||||||
$: templates = getTemplates($store, $backendUiStore.models)
|
$: templates = getTemplates($store, $backendUiStore.tables)
|
||||||
|
|
||||||
$: route = !route && $store.screens.length === 0 ? "*" : route
|
$: route = !route && $store.screens.length === 0 ? "*" : route
|
||||||
|
|
||||||
|
|
|
@ -37,7 +37,7 @@
|
||||||
componentInstanceId: $store.currentComponentInfo._id,
|
componentInstanceId: $store.currentComponentInfo._id,
|
||||||
components: $store.components,
|
components: $store.components,
|
||||||
screen: $store.currentPreviewItem,
|
screen: $store.currentPreviewItem,
|
||||||
models: $backendUiStore.models,
|
tables: $backendUiStore.tables,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -31,7 +31,7 @@
|
||||||
componentInstanceId: $store.currentComponentInfo._id,
|
componentInstanceId: $store.currentComponentInfo._id,
|
||||||
components: $store.components,
|
components: $store.components,
|
||||||
screen: $store.currentPreviewItem,
|
screen: $store.currentPreviewItem,
|
||||||
models: $backendUiStore.models,
|
tables: $backendUiStore.tables,
|
||||||
})
|
})
|
||||||
|
|
||||||
const detailScreens = $store.screens.filter(screen =>
|
const detailScreens = $store.screens.filter(screen =>
|
||||||
|
@ -43,11 +43,11 @@
|
||||||
if (
|
if (
|
||||||
p.type === "context" &&
|
p.type === "context" &&
|
||||||
p.runtimeBinding.endsWith("._id") &&
|
p.runtimeBinding.endsWith("._id") &&
|
||||||
p.model
|
p.table
|
||||||
) {
|
) {
|
||||||
const modelId =
|
const tableId =
|
||||||
typeof p.model === "string" ? p.model : p.model.modelId
|
typeof p.table === "string" ? p.table : p.table.tableId
|
||||||
return modelId === detailScreen.props.model
|
return tableId === detailScreen.props.table
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
|
|
|
@ -8,8 +8,8 @@
|
||||||
<div>
|
<div>
|
||||||
<Select thin secondary wide on:change {value}>
|
<Select thin secondary wide on:change {value}>
|
||||||
<option value="">Choose a table</option>
|
<option value="">Choose a table</option>
|
||||||
{#each $backendUiStore.models as model}
|
{#each $backendUiStore.tables as table}
|
||||||
<option value={model._id}>{model.name}</option>
|
<option value={table._id}>{table.name}</option>
|
||||||
{/each}
|
{/each}
|
||||||
</Select>
|
</Select>
|
||||||
</div>
|
</div>
|
|
@ -7,20 +7,20 @@
|
||||||
export let value = ""
|
export let value = ""
|
||||||
export let onChange = (val = {})
|
export let onChange = (val = {})
|
||||||
|
|
||||||
const models = $backendUiStore.models
|
const tables = $backendUiStore.tables
|
||||||
|
|
||||||
let options = []
|
let options = []
|
||||||
|
|
||||||
$: model = componentInstance.datasource
|
$: table = componentInstance.datasource
|
||||||
? models.find(m => m._id === componentInstance.datasource.modelId)
|
? tables.find(m => m._id === componentInstance.datasource.tableId)
|
||||||
: null
|
: null
|
||||||
|
|
||||||
$: type = componentInstance.datasource.type
|
$: type = componentInstance.datasource.type
|
||||||
$: if (model) {
|
$: if (table) {
|
||||||
options =
|
options =
|
||||||
type === "model" || type === "link"
|
type === "table" || type === "link"
|
||||||
? Object.keys(model.schema)
|
? Object.keys(table.schema)
|
||||||
: Object.keys(model.views[componentInstance.datasource.name].schema)
|
: Object.keys(table.views[componentInstance.datasource.name].schema)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -14,14 +14,14 @@
|
||||||
dropdownRight.hide()
|
dropdownRight.hide()
|
||||||
}
|
}
|
||||||
|
|
||||||
$: models = $backendUiStore.models.map(m => ({
|
$: tables = $backendUiStore.tables.map(m => ({
|
||||||
label: m.name,
|
label: m.name,
|
||||||
name: `all_${m._id}`,
|
name: `all_${m._id}`,
|
||||||
modelId: m._id,
|
tableId: m._id,
|
||||||
type: "model",
|
type: "table",
|
||||||
}))
|
}))
|
||||||
|
|
||||||
$: views = $backendUiStore.models.reduce((acc, cur) => {
|
$: views = $backendUiStore.tables.reduce((acc, cur) => {
|
||||||
let viewsArr = Object.entries(cur.views).map(([key, value]) => ({
|
let viewsArr = Object.entries(cur.views).map(([key, value]) => ({
|
||||||
label: key,
|
label: key,
|
||||||
name: key,
|
name: key,
|
||||||
|
@ -35,7 +35,7 @@
|
||||||
componentInstanceId: $store.currentComponentInfo._id,
|
componentInstanceId: $store.currentComponentInfo._id,
|
||||||
components: $store.components,
|
components: $store.components,
|
||||||
screen: $store.currentPreviewItem,
|
screen: $store.currentPreviewItem,
|
||||||
models: $backendUiStore.models,
|
tables: $backendUiStore.tables,
|
||||||
})
|
})
|
||||||
|
|
||||||
$: links = bindableProperties
|
$: links = bindableProperties
|
||||||
|
@ -43,8 +43,8 @@
|
||||||
.map(property => ({
|
.map(property => ({
|
||||||
label: property.readableBinding,
|
label: property.readableBinding,
|
||||||
fieldName: property.fieldSchema.name,
|
fieldName: property.fieldSchema.name,
|
||||||
name: `all_${property.fieldSchema.modelId}`,
|
name: `all_${property.fieldSchema.tableId}`,
|
||||||
modelId: property.fieldSchema.modelId,
|
tableId: property.fieldSchema.tableId,
|
||||||
type: "link",
|
type: "link",
|
||||||
}))
|
}))
|
||||||
</script>
|
</script>
|
||||||
|
@ -53,7 +53,7 @@
|
||||||
class="dropdownbutton"
|
class="dropdownbutton"
|
||||||
bind:this={anchorRight}
|
bind:this={anchorRight}
|
||||||
on:click={dropdownRight.show}>
|
on:click={dropdownRight.show}>
|
||||||
<span>{value.label ? value.label : 'Model / View'}</span>
|
<span>{value.label ? value.label : 'Table / View'}</span>
|
||||||
<Icon name="arrowdown" />
|
<Icon name="arrowdown" />
|
||||||
</div>
|
</div>
|
||||||
<DropdownMenu bind:this={dropdownRight} anchor={anchorRight}>
|
<DropdownMenu bind:this={dropdownRight} anchor={anchorRight}>
|
||||||
|
@ -62,11 +62,11 @@
|
||||||
<Heading extraSmall>Tables</Heading>
|
<Heading extraSmall>Tables</Heading>
|
||||||
</div>
|
</div>
|
||||||
<ul>
|
<ul>
|
||||||
{#each models as model}
|
{#each tables as table}
|
||||||
<li
|
<li
|
||||||
class:selected={value === model}
|
class:selected={value === table}
|
||||||
on:click={() => handleSelected(model)}>
|
on:click={() => handleSelected(table)}>
|
||||||
{model.label}
|
{table.label}
|
||||||
</li>
|
</li>
|
||||||
{/each}
|
{/each}
|
||||||
</ul>
|
</ul>
|
|
@ -19,7 +19,7 @@ export const TYPE_MAP = {
|
||||||
"##bbstate": "",
|
"##bbstate": "",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
models: {
|
tables: {
|
||||||
default: {},
|
default: {},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
import Input from "./PropertyPanelControls/Input.svelte"
|
import Input from "./PropertyPanelControls/Input.svelte"
|
||||||
import OptionSelect from "./OptionSelect.svelte"
|
import OptionSelect from "./OptionSelect.svelte"
|
||||||
import Checkbox from "../common/Checkbox.svelte"
|
import Checkbox from "../common/Checkbox.svelte"
|
||||||
import ModelSelect from "components/userInterface/ModelSelect.svelte"
|
import TableSelect from "components/userInterface/TableSelect.svelte"
|
||||||
import ModelViewSelect from "components/userInterface/ModelViewSelect.svelte"
|
import TableViewSelect from "components/userInterface/TableViewSelect.svelte"
|
||||||
import ModelViewFieldSelect from "components/userInterface/ModelViewFieldSelect.svelte"
|
import TableViewFieldSelect from "components/userInterface/TableViewFieldSelect.svelte"
|
||||||
import Event from "components/userInterface/EventsEditor/EventPropertyControl.svelte"
|
import Event from "components/userInterface/EventsEditor/EventPropertyControl.svelte"
|
||||||
import ScreenSelect from "components/userInterface/ScreenSelect.svelte"
|
import ScreenSelect from "components/userInterface/ScreenSelect.svelte"
|
||||||
import { IconSelect } from "components/userInterface/IconSelect"
|
import { IconSelect } from "components/userInterface/IconSelect"
|
||||||
|
@ -299,7 +299,7 @@ export default {
|
||||||
{
|
{
|
||||||
name: "List",
|
name: "List",
|
||||||
_component: "@budibase/standard-components/list",
|
_component: "@budibase/standard-components/list",
|
||||||
description: "Renders all children once per record, of a given table",
|
description: "Renders all children once per row, of a given table",
|
||||||
icon: "ri-file-list-line",
|
icon: "ri-file-list-line",
|
||||||
properties: {
|
properties: {
|
||||||
design: { ...all },
|
design: { ...all },
|
||||||
|
@ -307,7 +307,7 @@ export default {
|
||||||
{
|
{
|
||||||
label: "Data",
|
label: "Data",
|
||||||
key: "datasource",
|
key: "datasource",
|
||||||
control: ModelViewSelect,
|
control: TableViewSelect,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
@ -325,7 +325,7 @@ export default {
|
||||||
{
|
{
|
||||||
label: "Source",
|
label: "Source",
|
||||||
key: "datasource",
|
key: "datasource",
|
||||||
control: ModelViewSelect,
|
control: TableViewSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Editable",
|
label: "Editable",
|
||||||
|
@ -589,7 +589,7 @@ export default {
|
||||||
{
|
{
|
||||||
label: "Data",
|
label: "Data",
|
||||||
key: "datasource",
|
key: "datasource",
|
||||||
control: ModelViewSelect,
|
control: TableViewSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Stripe Color",
|
label: "Stripe Color",
|
||||||
|
@ -615,7 +615,7 @@ export default {
|
||||||
control: Colorpicker,
|
control: Colorpicker,
|
||||||
defaultValue: "#FFFFFF",
|
defaultValue: "#FFFFFF",
|
||||||
},
|
},
|
||||||
{ label: "Table", key: "model", control: ModelSelect },
|
{ label: "Table", key: "table", control: TableSelect },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
children: [],
|
children: [],
|
||||||
|
@ -661,19 +661,19 @@ export default {
|
||||||
{
|
{
|
||||||
label: "Data",
|
label: "Data",
|
||||||
key: "datasource",
|
key: "datasource",
|
||||||
control: ModelViewSelect,
|
control: TableViewSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Name Field",
|
label: "Name Field",
|
||||||
key: "nameKey",
|
key: "nameKey",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Value Field",
|
label: "Value Field",
|
||||||
key: "valueKey",
|
key: "valueKey",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Animate Chart",
|
label: "Animate Chart",
|
||||||
|
@ -755,19 +755,19 @@ export default {
|
||||||
{
|
{
|
||||||
label: "Data",
|
label: "Data",
|
||||||
key: "datasource",
|
key: "datasource",
|
||||||
control: ModelViewSelect,
|
control: TableViewSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Name Label",
|
label: "Name Label",
|
||||||
key: "nameLabel",
|
key: "nameLabel",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Value Label",
|
label: "Value Label",
|
||||||
key: "valueLabel",
|
key: "valueLabel",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Y Axis Label",
|
label: "Y Axis Label",
|
||||||
|
@ -869,25 +869,25 @@ export default {
|
||||||
{
|
{
|
||||||
label: "Data",
|
label: "Data",
|
||||||
key: "datasource",
|
key: "datasource",
|
||||||
control: ModelViewSelect,
|
control: TableViewSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Name Label",
|
label: "Name Label",
|
||||||
key: "nameLabel",
|
key: "nameLabel",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Group Label",
|
label: "Group Label",
|
||||||
key: "groupLabel",
|
key: "groupLabel",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Value Label",
|
label: "Value Label",
|
||||||
key: "valueLabel",
|
key: "valueLabel",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Color",
|
label: "Color",
|
||||||
|
@ -972,25 +972,25 @@ export default {
|
||||||
{
|
{
|
||||||
label: "Data",
|
label: "Data",
|
||||||
key: "datasource",
|
key: "datasource",
|
||||||
control: ModelViewSelect,
|
control: TableViewSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Value Label",
|
label: "Value Label",
|
||||||
key: "valueLabel",
|
key: "valueLabel",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Topic Label",
|
label: "Topic Label",
|
||||||
key: "topicLabel",
|
key: "topicLabel",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Date Label",
|
label: "Date Label",
|
||||||
key: "dateLabel",
|
key: "dateLabel",
|
||||||
dependsOn: "datasource",
|
dependsOn: "datasource",
|
||||||
control: ModelViewFieldSelect,
|
control: TableViewFieldSelect,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Colors",
|
label: "Colors",
|
||||||
|
@ -1137,7 +1137,7 @@ export default {
|
||||||
// icon: "ri-file-list-line",
|
// icon: "ri-file-list-line",
|
||||||
// properties: {
|
// properties: {
|
||||||
// design: { ...all },
|
// design: { ...all },
|
||||||
// settings: [{ label: "Table", key: "model", control: ModelSelect }],
|
// settings: [{ label: "Table", key: "table", control: TableSelect }],
|
||||||
// },
|
// },
|
||||||
// children: [],
|
// children: [],
|
||||||
// },
|
// },
|
||||||
|
@ -1145,11 +1145,11 @@ export default {
|
||||||
name: "Row Detail",
|
name: "Row Detail",
|
||||||
_component: "@budibase/standard-components/rowdetail",
|
_component: "@budibase/standard-components/rowdetail",
|
||||||
description:
|
description:
|
||||||
"Loads a record, using an id from the URL, which can be used with {{ context }}, in children",
|
"Loads a row, using an id from the URL, which can be used with {{ context }}, in children",
|
||||||
icon: "ri-profile-line",
|
icon: "ri-profile-line",
|
||||||
properties: {
|
properties: {
|
||||||
design: { ...all },
|
design: { ...all },
|
||||||
settings: [{ label: "Table", key: "model", control: ModelSelect }],
|
settings: [{ label: "Table", key: "table", control: TableSelect }],
|
||||||
},
|
},
|
||||||
children: [],
|
children: [],
|
||||||
},
|
},
|
||||||
|
@ -1157,11 +1157,11 @@ export default {
|
||||||
name: "New Row",
|
name: "New Row",
|
||||||
_component: "@budibase/standard-components/newrow",
|
_component: "@budibase/standard-components/newrow",
|
||||||
description:
|
description:
|
||||||
"Sets up a new record for creation, which can be used with {{ context }}, in children",
|
"Sets up a new row for creation, which can be used with {{ context }}, in children",
|
||||||
icon: "ri-profile-line",
|
icon: "ri-profile-line",
|
||||||
properties: {
|
properties: {
|
||||||
design: { ...all },
|
design: { ...all },
|
||||||
settings: [{ label: "Table", key: "model", control: ModelSelect }],
|
settings: [{ label: "Table", key: "table", control: TableSelect }],
|
||||||
},
|
},
|
||||||
children: [],
|
children: [],
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
<script>
|
<script>
|
||||||
import { store, backendUiStore } from "builderStore"
|
import { store, backendUiStore } from "builderStore"
|
||||||
import * as api from "components/backend/DataTable/api"
|
import * as api from "components/backend/DataTable/api"
|
||||||
import ModelNavigator from "components/backend/ModelNavigator/ModelNavigator.svelte"
|
import TableNavigator from "components/backend/TableNavigator/TableNavigator.svelte"
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<!-- routify:options index=1 -->
|
<!-- routify:options index=1 -->
|
||||||
<div class="root">
|
<div class="root">
|
||||||
<div class="nav">
|
<div class="nav">
|
||||||
<ModelNavigator />
|
<TableNavigator />
|
||||||
</div>
|
</div>
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<slot />
|
<slot />
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<script>
|
<script>
|
||||||
import { goto } from "@sveltech/routify"
|
import { goto } from "@sveltech/routify"
|
||||||
$goto("../model")
|
$goto("../table")
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<!-- routify:options index=false -->
|
<!-- routify:options index=false -->
|
||||||
|
|
|
@ -1,15 +0,0 @@
|
||||||
<script>
|
|
||||||
import { params } from "@sveltech/routify"
|
|
||||||
import { backendUiStore } from "builderStore"
|
|
||||||
|
|
||||||
if ($params.selectedModel) {
|
|
||||||
const model = $backendUiStore.models.find(
|
|
||||||
m => m._id === $params.selectedModel
|
|
||||||
)
|
|
||||||
if (model) {
|
|
||||||
backendUiStore.actions.models.select(model)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<slot />
|
|
|
@ -1,32 +0,0 @@
|
||||||
<script>
|
|
||||||
import { backendUiStore } from "builderStore"
|
|
||||||
import { goto, leftover } from "@sveltech/routify"
|
|
||||||
import { onMount } from "svelte"
|
|
||||||
|
|
||||||
async function selectModel(model) {
|
|
||||||
backendUiStore.actions.models.select(model)
|
|
||||||
}
|
|
||||||
|
|
||||||
onMount(async () => {
|
|
||||||
// navigate to first model in list, if not already selected
|
|
||||||
// and this is the final url (i.e. no selectedModel)
|
|
||||||
if (
|
|
||||||
!$leftover &&
|
|
||||||
$backendUiStore.models.length > 0 &&
|
|
||||||
(!$backendUiStore.selectedModel || !$backendUiStore.selectedModel._id)
|
|
||||||
) {
|
|
||||||
$goto(`./${$backendUiStore.models[0]._id}`)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<div class="root">
|
|
||||||
<slot />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.root {
|
|
||||||
height: 100%;
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
</style>
|
|
|
@ -1,35 +0,0 @@
|
||||||
<script>
|
|
||||||
import { store, backendUiStore } from "builderStore"
|
|
||||||
import { goto, leftover } from "@sveltech/routify"
|
|
||||||
import { onMount } from "svelte"
|
|
||||||
|
|
||||||
async function selectModel(model) {
|
|
||||||
backendUiStore.actions.models.select(model)
|
|
||||||
}
|
|
||||||
|
|
||||||
onMount(async () => {
|
|
||||||
// navigate to first model in list, if not already selected
|
|
||||||
// and this is the final url (i.e. no selectedModel)
|
|
||||||
if (
|
|
||||||
!$leftover &&
|
|
||||||
$backendUiStore.models.length > 0 &&
|
|
||||||
(!$backendUiStore.selectedModel || !$backendUiStore.selectedModel._id)
|
|
||||||
) {
|
|
||||||
// this file routes as .../models/index, so, go up one.
|
|
||||||
$goto(`../${$backendUiStore.models[0]._id}`)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
{#if $backendUiStore.models.length === 0}
|
|
||||||
<i>Create your first table to start building</i>
|
|
||||||
{:else}
|
|
||||||
<i>Select a table to edit</i>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<style>
|
|
||||||
i {
|
|
||||||
font-size: var(--font-size-xl);
|
|
||||||
color: var(--grey-4);
|
|
||||||
}
|
|
||||||
</style>
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
<script>
|
||||||
|
import { params } from "@sveltech/routify"
|
||||||
|
import { backendUiStore } from "builderStore"
|
||||||
|
|
||||||
|
if ($params.selectedTable) {
|
||||||
|
const table = $backendUiStore.tables.find(
|
||||||
|
m => m._id === $params.selectedTable
|
||||||
|
)
|
||||||
|
if (table) {
|
||||||
|
backendUiStore.actions.tables.select(table)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<slot />
|
|
@ -1,12 +1,12 @@
|
||||||
<script>
|
<script>
|
||||||
import ModelDataTable from "components/backend/DataTable/ModelDataTable.svelte"
|
import TableDataTable from "components/backend/DataTable/DataTable.svelte"
|
||||||
import { backendUiStore } from "builderStore"
|
import { backendUiStore } from "builderStore"
|
||||||
|
|
||||||
$: selectedModel = $backendUiStore.selectedModel
|
$: selectedTable = $backendUiStore.selectedTable
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if $backendUiStore.selectedDatabase._id && selectedModel.name}
|
{#if $backendUiStore.selectedDatabase._id && selectedTable.name}
|
||||||
<ModelDataTable />
|
<TableDataTable />
|
||||||
{:else}
|
{:else}
|
||||||
<i>Create your first table to start building</i>
|
<i>Create your first table to start building</i>
|
||||||
{/if}
|
{/if}
|
|
@ -4,6 +4,6 @@
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<RelationshipDataTable
|
<RelationshipDataTable
|
||||||
modelId={$params.selectedModel}
|
tableId={$params.selectedTable}
|
||||||
recordId={$params.selectedRecord}
|
rowId={$params.selectedRow}
|
||||||
fieldName={decodeURI($params.selectedField)} />
|
fieldName={decodeURI($params.selectedField)} />
|
|
@ -0,0 +1,32 @@
|
||||||
|
<script>
|
||||||
|
import { backendUiStore } from "builderStore"
|
||||||
|
import { goto, leftover } from "@sveltech/routify"
|
||||||
|
import { onMount } from "svelte"
|
||||||
|
|
||||||
|
async function selectTable(table) {
|
||||||
|
backendUiStore.actions.tables.select(table)
|
||||||
|
}
|
||||||
|
|
||||||
|
onMount(async () => {
|
||||||
|
// navigate to first table in list, if not already selected
|
||||||
|
// and this is the final url (i.e. no selectedTable)
|
||||||
|
if (
|
||||||
|
!$leftover &&
|
||||||
|
$backendUiStore.tables.length > 0 &&
|
||||||
|
(!$backendUiStore.selectedTable || !$backendUiStore.selectedTable._id)
|
||||||
|
) {
|
||||||
|
$goto(`./${$backendUiStore.tables[0]._id}`)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="root">
|
||||||
|
<slot />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.root {
|
||||||
|
height: 100%;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,35 @@
|
||||||
|
<script>
|
||||||
|
import { store, backendUiStore } from "builderStore"
|
||||||
|
import { goto, leftover } from "@sveltech/routify"
|
||||||
|
import { onMount } from "svelte"
|
||||||
|
|
||||||
|
async function selectTable(table) {
|
||||||
|
backendUiStore.actions.tables.select(table)
|
||||||
|
}
|
||||||
|
|
||||||
|
onMount(async () => {
|
||||||
|
// navigate to first table in list, if not already selected
|
||||||
|
// and this is the final url (i.e. no selectedTable)
|
||||||
|
if (
|
||||||
|
!$leftover &&
|
||||||
|
$backendUiStore.tables.length > 0 &&
|
||||||
|
(!$backendUiStore.selectedTable || !$backendUiStore.selectedTable._id)
|
||||||
|
) {
|
||||||
|
// this file routes as .../tables/index, so, go up one.
|
||||||
|
$goto(`../${$backendUiStore.tables[0]._id}`)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{#if $backendUiStore.tables.length === 0}
|
||||||
|
<i>Create your first table to start building</i>
|
||||||
|
{:else}
|
||||||
|
<i>Select a table to edit</i>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<style>
|
||||||
|
i {
|
||||||
|
font-size: var(--font-size-xl);
|
||||||
|
color: var(--grey-4);
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -5,9 +5,9 @@
|
||||||
if ($params.selectedView) {
|
if ($params.selectedView) {
|
||||||
let view
|
let view
|
||||||
const viewName = decodeURI($params.selectedView)
|
const viewName = decodeURI($params.selectedView)
|
||||||
for (let model of $backendUiStore.models) {
|
for (let table of $backendUiStore.tables) {
|
||||||
if (model.views && model.views[viewName]) {
|
if (table.views && table.views[viewName]) {
|
||||||
view = model.views[viewName]
|
view = table.views[viewName]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (view) {
|
if (view) {
|
||||||
|
|
|
@ -25,7 +25,7 @@ describe("fetch bindable properties", () => {
|
||||||
expect(componentBinding).not.toBeDefined()
|
expect(componentBinding).not.toBeDefined()
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should return model schema, when inside a context", () => {
|
it("should return table schema, when inside a context", () => {
|
||||||
const result = fetchBindableProperties({
|
const result = fetchBindableProperties({
|
||||||
componentInstanceId: "list-item-input-id",
|
componentInstanceId: "list-item-input-id",
|
||||||
...testData(),
|
...testData(),
|
||||||
|
@ -40,42 +40,42 @@ describe("fetch bindable properties", () => {
|
||||||
b => b.runtimeBinding === "data.name"
|
b => b.runtimeBinding === "data.name"
|
||||||
)
|
)
|
||||||
expect(namebinding).toBeDefined()
|
expect(namebinding).toBeDefined()
|
||||||
expect(namebinding.readableBinding).toBe("list-name.Test Model.name")
|
expect(namebinding.readableBinding).toBe("list-name.Test Table.name")
|
||||||
|
|
||||||
const descriptionbinding = contextBindings.find(
|
const descriptionbinding = contextBindings.find(
|
||||||
b => b.runtimeBinding === "data.description"
|
b => b.runtimeBinding === "data.description"
|
||||||
)
|
)
|
||||||
expect(descriptionbinding).toBeDefined()
|
expect(descriptionbinding).toBeDefined()
|
||||||
expect(descriptionbinding.readableBinding).toBe(
|
expect(descriptionbinding.readableBinding).toBe(
|
||||||
"list-name.Test Model.description"
|
"list-name.Test Table.description"
|
||||||
)
|
)
|
||||||
|
|
||||||
const idbinding = contextBindings.find(b => b.runtimeBinding === "data._id")
|
const idbinding = contextBindings.find(b => b.runtimeBinding === "data._id")
|
||||||
expect(idbinding).toBeDefined()
|
expect(idbinding).toBeDefined()
|
||||||
expect(idbinding.readableBinding).toBe("list-name.Test Model._id")
|
expect(idbinding.readableBinding).toBe("list-name.Test Table._id")
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should return model schema, for grantparent context", () => {
|
it("should return table schema, for grantparent context", () => {
|
||||||
const result = fetchBindableProperties({
|
const result = fetchBindableProperties({
|
||||||
componentInstanceId: "child-list-item-input-id",
|
componentInstanceId: "child-list-item-input-id",
|
||||||
...testData(),
|
...testData(),
|
||||||
})
|
})
|
||||||
const contextBindings = result.filter(r => r.type === "context")
|
const contextBindings = result.filter(r => r.type === "context")
|
||||||
// 2 fields + _id + _rev ... x 2 models
|
// 2 fields + _id + _rev ... x 2 tables
|
||||||
expect(contextBindings.length).toBe(8)
|
expect(contextBindings.length).toBe(8)
|
||||||
|
|
||||||
const namebinding_parent = contextBindings.find(
|
const namebinding_parent = contextBindings.find(
|
||||||
b => b.runtimeBinding === "parent.data.name"
|
b => b.runtimeBinding === "parent.data.name"
|
||||||
)
|
)
|
||||||
expect(namebinding_parent).toBeDefined()
|
expect(namebinding_parent).toBeDefined()
|
||||||
expect(namebinding_parent.readableBinding).toBe("list-name.Test Model.name")
|
expect(namebinding_parent.readableBinding).toBe("list-name.Test Table.name")
|
||||||
|
|
||||||
const descriptionbinding_parent = contextBindings.find(
|
const descriptionbinding_parent = contextBindings.find(
|
||||||
b => b.runtimeBinding === "parent.data.description"
|
b => b.runtimeBinding === "parent.data.description"
|
||||||
)
|
)
|
||||||
expect(descriptionbinding_parent).toBeDefined()
|
expect(descriptionbinding_parent).toBeDefined()
|
||||||
expect(descriptionbinding_parent.readableBinding).toBe(
|
expect(descriptionbinding_parent.readableBinding).toBe(
|
||||||
"list-name.Test Model.description"
|
"list-name.Test Table.description"
|
||||||
)
|
)
|
||||||
|
|
||||||
const namebinding_own = contextBindings.find(
|
const namebinding_own = contextBindings.find(
|
||||||
|
@ -83,7 +83,7 @@ describe("fetch bindable properties", () => {
|
||||||
)
|
)
|
||||||
expect(namebinding_own).toBeDefined()
|
expect(namebinding_own).toBeDefined()
|
||||||
expect(namebinding_own.readableBinding).toBe(
|
expect(namebinding_own.readableBinding).toBe(
|
||||||
"child-list-name.Test Model.name"
|
"child-list-name.Test Table.name"
|
||||||
)
|
)
|
||||||
|
|
||||||
const descriptionbinding_own = contextBindings.find(
|
const descriptionbinding_own = contextBindings.find(
|
||||||
|
@ -91,7 +91,7 @@ describe("fetch bindable properties", () => {
|
||||||
)
|
)
|
||||||
expect(descriptionbinding_own).toBeDefined()
|
expect(descriptionbinding_own).toBeDefined()
|
||||||
expect(descriptionbinding_own.readableBinding).toBe(
|
expect(descriptionbinding_own.readableBinding).toBe(
|
||||||
"child-list-name.Test Model.description"
|
"child-list-name.Test Table.description"
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -157,11 +157,11 @@ const testData = () => {
|
||||||
_id: "list-id",
|
_id: "list-id",
|
||||||
_component: "@budibase/standard-components/list",
|
_component: "@budibase/standard-components/list",
|
||||||
_instanceName: "list-name",
|
_instanceName: "list-name",
|
||||||
model: {
|
table: {
|
||||||
type: "model",
|
type: "table",
|
||||||
modelId: "test-model-id",
|
tableId: "test-table-id",
|
||||||
label: "Test Model",
|
label: "Test Table",
|
||||||
name: "all_test-model-id",
|
name: "all_test-table-id",
|
||||||
},
|
},
|
||||||
_children: [
|
_children: [
|
||||||
{
|
{
|
||||||
|
@ -180,11 +180,11 @@ const testData = () => {
|
||||||
_id: "child-list-id",
|
_id: "child-list-id",
|
||||||
_component: "@budibase/standard-components/list",
|
_component: "@budibase/standard-components/list",
|
||||||
_instanceName: "child-list-name",
|
_instanceName: "child-list-name",
|
||||||
model: {
|
table: {
|
||||||
type: "model",
|
type: "table",
|
||||||
modelId: "test-model-id",
|
tableId: "test-table-id",
|
||||||
label: "Test Model",
|
label: "Test Table",
|
||||||
name: "all_test-model-id",
|
name: "all_test-table-id",
|
||||||
},
|
},
|
||||||
_children: [
|
_children: [
|
||||||
{
|
{
|
||||||
|
@ -207,10 +207,10 @@ const testData = () => {
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
const models = [
|
const tables = [
|
||||||
{
|
{
|
||||||
_id: "test-model-id",
|
_id: "test-table-id",
|
||||||
name: "Test Model",
|
name: "Test Table",
|
||||||
schema: {
|
schema: {
|
||||||
name: {
|
name: {
|
||||||
type: "string",
|
type: "string",
|
||||||
|
@ -227,9 +227,9 @@ const testData = () => {
|
||||||
props: {},
|
props: {},
|
||||||
},
|
},
|
||||||
"@budibase/standard-components/list": {
|
"@budibase/standard-components/list": {
|
||||||
context: "model",
|
context: "table",
|
||||||
props: {
|
props: {
|
||||||
model: "string",
|
table: "string",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
"@budibase/standard-components/input": {
|
"@budibase/standard-components/input": {
|
||||||
|
@ -245,5 +245,5 @@ const testData = () => {
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
return { screen, models, components }
|
return { screen, tables, components }
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,8 +29,8 @@ export const componentsAndScreens = () => ({
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
_instanceName: "Record View",
|
_instanceName: "Row View",
|
||||||
tags: ["record"],
|
tags: ["row"],
|
||||||
props: {
|
props: {
|
||||||
data: "state",
|
data: "state",
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "budibase",
|
"name": "budibase",
|
||||||
"version": "0.2.0",
|
"version": "0.2.1",
|
||||||
"description": "Budibase CLI",
|
"description": "Budibase CLI",
|
||||||
"repository": "https://github.com/Budibase/Budibase",
|
"repository": "https://github.com/Budibase/Budibase",
|
||||||
"homepage": "https://www.budibase.com",
|
"homepage": "https://www.budibase.com",
|
||||||
|
@ -17,7 +17,7 @@
|
||||||
"author": "Budibase",
|
"author": "Budibase",
|
||||||
"license": "AGPL-3.0-or-later",
|
"license": "AGPL-3.0-or-later",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@budibase/server": "^0.2.0",
|
"@budibase/server": "^0.2.1",
|
||||||
"@inquirer/password": "^0.0.6-alpha.0",
|
"@inquirer/password": "^0.0.6-alpha.0",
|
||||||
"chalk": "^2.4.2",
|
"chalk": "^2.4.2",
|
||||||
"dotenv": "^8.2.0",
|
"dotenv": "^8.2.0",
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "@budibase/client",
|
"name": "@budibase/client",
|
||||||
"version": "0.2.0",
|
"version": "0.2.1",
|
||||||
"license": "MPL-2.0",
|
"license": "MPL-2.0",
|
||||||
"main": "dist/budibase-client.js",
|
"main": "dist/budibase-client.js",
|
||||||
"module": "dist/budibase-client.esm.mjs",
|
"module": "dist/budibase-client.esm.mjs",
|
||||||
|
|
|
@ -52,27 +52,27 @@ const apiOpts = {
|
||||||
delete: del,
|
delete: del,
|
||||||
}
|
}
|
||||||
|
|
||||||
const saveRecord = async (params, state) =>
|
const saveRow = async (params, state) =>
|
||||||
await post({
|
await post({
|
||||||
url: `/api/${params.modelId}/records`,
|
url: `/api/${params.tableId}/rows`,
|
||||||
body: makeRecordRequestBody(params, state),
|
body: makeRowRequestBody(params, state),
|
||||||
})
|
})
|
||||||
|
|
||||||
const updateRecord = async (params, state) => {
|
const updateRow = async (params, state) => {
|
||||||
const record = makeRecordRequestBody(params, state)
|
const row = makeRowRequestBody(params, state)
|
||||||
record._id = params._id
|
row._id = params._id
|
||||||
await patch({
|
await patch({
|
||||||
url: `/api/${params.modelId}/records/${params._id}`,
|
url: `/api/${params.tableId}/rows/${params._id}`,
|
||||||
body: record,
|
body: row,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
const makeRecordRequestBody = (parameters, state) => {
|
const makeRowRequestBody = (parameters, state) => {
|
||||||
// start with the record thats currently in context
|
// start with the row thats currently in context
|
||||||
const body = { ...(state.data || {}) }
|
const body = { ...(state.data || {}) }
|
||||||
|
|
||||||
// dont send the model
|
// dont send the table
|
||||||
if (body._model) delete body._model
|
if (body._table) delete body._table
|
||||||
|
|
||||||
// then override with supplied parameters
|
// then override with supplied parameters
|
||||||
for (let fieldName in parameters.fields) {
|
for (let fieldName in parameters.fields) {
|
||||||
|
@ -101,6 +101,6 @@ const makeRecordRequestBody = (parameters, state) => {
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
authenticate: authenticate(apiOpts),
|
authenticate: authenticate(apiOpts),
|
||||||
saveRecord,
|
saveRow,
|
||||||
updateRecord,
|
updateRow,
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,8 +6,8 @@ export const EVENT_TYPE_MEMBER_NAME = "##eventHandlerType"
|
||||||
export const eventHandlers = routeTo => {
|
export const eventHandlers = routeTo => {
|
||||||
const handlers = {
|
const handlers = {
|
||||||
"Navigate To": param => routeTo(param && param.url),
|
"Navigate To": param => routeTo(param && param.url),
|
||||||
"Update Record": api.updateRecord,
|
"Update Row": api.updateRow,
|
||||||
"Save Record": api.saveRecord,
|
"Save Row": api.saveRow,
|
||||||
"Trigger Workflow": api.triggerWorkflow,
|
"Trigger Workflow": api.triggerWorkflow,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -80,7 +80,7 @@
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"name": "Jest - Models",
|
"name": "Jest - Models",
|
||||||
"program": "${workspaceFolder}/node_modules/.bin/jest",
|
"program": "${workspaceFolder}/node_modules/.bin/jest",
|
||||||
"args": ["model.spec", "--runInBand"],
|
"args": ["table.spec", "--runInBand"],
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"internalConsoleOptions": "neverOpen",
|
"internalConsoleOptions": "neverOpen",
|
||||||
"disableOptimisticBPs": true,
|
"disableOptimisticBPs": true,
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "@budibase/server",
|
"name": "@budibase/server",
|
||||||
"version": "0.2.0",
|
"version": "0.2.1",
|
||||||
"description": "Budibase Web Server",
|
"description": "Budibase Web Server",
|
||||||
"main": "src/electron.js",
|
"main": "src/electron.js",
|
||||||
"repository": {
|
"repository": {
|
||||||
|
@ -42,7 +42,7 @@
|
||||||
"author": "Michael Shanks",
|
"author": "Michael Shanks",
|
||||||
"license": "AGPL-3.0-or-later",
|
"license": "AGPL-3.0-or-later",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@budibase/client": "^0.2.0",
|
"@budibase/client": "^0.2.1",
|
||||||
"@koa/router": "^8.0.0",
|
"@koa/router": "^8.0.0",
|
||||||
"@sendgrid/mail": "^7.1.1",
|
"@sendgrid/mail": "^7.1.1",
|
||||||
"@sentry/node": "^5.19.2",
|
"@sentry/node": "^5.19.2",
|
||||||
|
|
|
@ -24,14 +24,14 @@ async function run() {
|
||||||
quotaReset: Date.now() + 2592000000,
|
quotaReset: Date.now() + 2592000000,
|
||||||
usageQuota: {
|
usageQuota: {
|
||||||
automationRuns: 0,
|
automationRuns: 0,
|
||||||
records: 0,
|
rows: 0,
|
||||||
storage: 0,
|
storage: 0,
|
||||||
users: 0,
|
users: 0,
|
||||||
views: 0,
|
views: 0,
|
||||||
},
|
},
|
||||||
usageLimits: {
|
usageLimits: {
|
||||||
automationRuns: 10,
|
automationRuns: 10,
|
||||||
records: 10,
|
rows: 10,
|
||||||
storage: 1000,
|
storage: 1000,
|
||||||
users: 10,
|
users: 10,
|
||||||
views: 10,
|
views: 10,
|
||||||
|
@ -48,8 +48,8 @@ async function run() {
|
||||||
|
|
||||||
run()
|
run()
|
||||||
.then(() => {
|
.then(() => {
|
||||||
console.log("Records should have been created.")
|
console.log("Rows should have been created.")
|
||||||
})
|
})
|
||||||
.catch(err => {
|
.catch(err => {
|
||||||
console.error("Cannot create records - " + err)
|
console.error("Cannot create rows - " + err)
|
||||||
})
|
})
|
||||||
|
|
|
@ -39,9 +39,9 @@ async function replicateCouch({ instanceId, clientId, credentials }) {
|
||||||
async function getCurrentInstanceQuota(instanceId) {
|
async function getCurrentInstanceQuota(instanceId) {
|
||||||
const db = new PouchDB(instanceId)
|
const db = new PouchDB(instanceId)
|
||||||
|
|
||||||
const records = await db.allDocs({
|
const rows = await db.allDocs({
|
||||||
startkey: DocumentTypes.RECORD + SEPARATOR,
|
startkey: DocumentTypes.ROW + SEPARATOR,
|
||||||
endkey: DocumentTypes.RECORD + SEPARATOR + UNICODE_MAX,
|
endkey: DocumentTypes.ROW + SEPARATOR + UNICODE_MAX,
|
||||||
})
|
})
|
||||||
|
|
||||||
const users = await db.allDocs({
|
const users = await db.allDocs({
|
||||||
|
@ -49,13 +49,13 @@ async function getCurrentInstanceQuota(instanceId) {
|
||||||
endkey: DocumentTypes.USER + SEPARATOR + UNICODE_MAX,
|
endkey: DocumentTypes.USER + SEPARATOR + UNICODE_MAX,
|
||||||
})
|
})
|
||||||
|
|
||||||
const existingRecords = records.rows.length
|
const existingRows = rows.rows.length
|
||||||
const existingUsers = users.rows.length
|
const existingUsers = users.rows.length
|
||||||
|
|
||||||
const designDoc = await db.get("_design/database")
|
const designDoc = await db.get("_design/database")
|
||||||
|
|
||||||
return {
|
return {
|
||||||
records: existingRecords,
|
rows: existingRows,
|
||||||
users: existingUsers,
|
users: existingUsers,
|
||||||
views: Object.keys(designDoc.views).length,
|
views: Object.keys(designDoc.views).length,
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ const fs = require("fs")
|
||||||
const CouchDB = require("../../db")
|
const CouchDB = require("../../db")
|
||||||
const client = require("../../db/clientDb")
|
const client = require("../../db/clientDb")
|
||||||
const newid = require("../../db/newid")
|
const newid = require("../../db/newid")
|
||||||
const { createLinkView } = require("../../db/linkedRecords")
|
const { createLinkView } = require("../../db/linkedRows")
|
||||||
const { join } = require("../../utilities/centralPath")
|
const { join } = require("../../utilities/centralPath")
|
||||||
const { downloadTemplate } = require("../../utilities/templates")
|
const { downloadTemplate } = require("../../utilities/templates")
|
||||||
|
|
||||||
|
@ -27,7 +27,7 @@ exports.create = async function(ctx) {
|
||||||
// https://docs.couchdb.org/en/master/ddocs/views/collation.html#collation-specification
|
// https://docs.couchdb.org/en/master/ddocs/views/collation.html#collation-specification
|
||||||
views: {},
|
views: {},
|
||||||
})
|
})
|
||||||
// add view for linked records
|
// add view for linked rows
|
||||||
await createLinkView(instanceId)
|
await createLinkView(instanceId)
|
||||||
|
|
||||||
// Add the new instance under the app clientDB
|
// Add the new instance under the app clientDB
|
||||||
|
|
|
@ -1,146 +0,0 @@
|
||||||
const CouchDB = require("../../db")
|
|
||||||
const linkRecords = require("../../db/linkedRecords")
|
|
||||||
const csvParser = require("../../utilities/csvParser")
|
|
||||||
const {
|
|
||||||
getRecordParams,
|
|
||||||
getModelParams,
|
|
||||||
generateModelID,
|
|
||||||
generateRecordID,
|
|
||||||
} = require("../../db/utils")
|
|
||||||
|
|
||||||
exports.fetch = async function(ctx) {
|
|
||||||
const db = new CouchDB(ctx.user.instanceId)
|
|
||||||
const body = await db.allDocs(
|
|
||||||
getModelParams(null, {
|
|
||||||
include_docs: true,
|
|
||||||
})
|
|
||||||
)
|
|
||||||
ctx.body = body.rows.map(row => row.doc)
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.find = async function(ctx) {
|
|
||||||
const db = new CouchDB(ctx.user.instanceId)
|
|
||||||
ctx.body = await db.get(ctx.params.id)
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.save = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
const { dataImport, ...rest } = ctx.request.body
|
|
||||||
const modelToSave = {
|
|
||||||
type: "model",
|
|
||||||
_id: generateModelID(),
|
|
||||||
views: {},
|
|
||||||
...rest,
|
|
||||||
}
|
|
||||||
let renameDocs = []
|
|
||||||
|
|
||||||
// if the model obj had an _id then it will have been retrieved
|
|
||||||
const oldModel = ctx.preExisting
|
|
||||||
|
|
||||||
// rename record fields when table column is renamed
|
|
||||||
const { _rename } = modelToSave
|
|
||||||
if (_rename && modelToSave.schema[_rename.updated].type === "link") {
|
|
||||||
throw "Cannot rename a linked field."
|
|
||||||
} else if (_rename && modelToSave.primaryDisplay === _rename.old) {
|
|
||||||
throw "Cannot rename the primary display field."
|
|
||||||
} else if (_rename) {
|
|
||||||
const records = await db.allDocs(
|
|
||||||
getRecordParams(modelToSave._id, null, {
|
|
||||||
include_docs: true,
|
|
||||||
})
|
|
||||||
)
|
|
||||||
renameDocs = records.rows.map(({ doc }) => {
|
|
||||||
doc[_rename.updated] = doc[_rename.old]
|
|
||||||
delete doc[_rename.old]
|
|
||||||
return doc
|
|
||||||
})
|
|
||||||
delete modelToSave._rename
|
|
||||||
}
|
|
||||||
|
|
||||||
// update schema of non-statistics views when new columns are added
|
|
||||||
for (let view in modelToSave.views) {
|
|
||||||
const modelView = modelToSave.views[view]
|
|
||||||
if (!modelView) continue
|
|
||||||
|
|
||||||
if (modelView.schema.group || modelView.schema.field) continue
|
|
||||||
modelView.schema = modelToSave.schema
|
|
||||||
}
|
|
||||||
|
|
||||||
// update linked records
|
|
||||||
await linkRecords.updateLinks({
|
|
||||||
instanceId,
|
|
||||||
eventType: oldModel
|
|
||||||
? linkRecords.EventType.MODEL_UPDATED
|
|
||||||
: linkRecords.EventType.MODEL_SAVE,
|
|
||||||
model: modelToSave,
|
|
||||||
oldModel: oldModel,
|
|
||||||
})
|
|
||||||
|
|
||||||
// don't perform any updates until relationships have been
|
|
||||||
// checked by the updateLinks function
|
|
||||||
if (renameDocs.length !== 0) {
|
|
||||||
await db.bulkDocs(renameDocs)
|
|
||||||
}
|
|
||||||
const result = await db.post(modelToSave)
|
|
||||||
modelToSave._rev = result.rev
|
|
||||||
|
|
||||||
ctx.eventEmitter &&
|
|
||||||
ctx.eventEmitter.emitModel(`model:save`, instanceId, modelToSave)
|
|
||||||
|
|
||||||
if (dataImport && dataImport.path) {
|
|
||||||
// Populate the table with records imported from CSV in a bulk update
|
|
||||||
const data = await csvParser.transform(dataImport)
|
|
||||||
|
|
||||||
for (let row of data) {
|
|
||||||
row._id = generateRecordID(modelToSave._id)
|
|
||||||
row.modelId = modelToSave._id
|
|
||||||
}
|
|
||||||
|
|
||||||
await db.bulkDocs(data)
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.status = 200
|
|
||||||
ctx.message = `Model ${ctx.request.body.name} saved successfully.`
|
|
||||||
ctx.body = modelToSave
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.destroy = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
const modelToDelete = await db.get(ctx.params.modelId)
|
|
||||||
|
|
||||||
// Delete all records for that model
|
|
||||||
const records = await db.allDocs(
|
|
||||||
getRecordParams(ctx.params.modelId, null, {
|
|
||||||
include_docs: true,
|
|
||||||
})
|
|
||||||
)
|
|
||||||
await db.bulkDocs(
|
|
||||||
records.rows.map(record => ({ ...record.doc, _deleted: true }))
|
|
||||||
)
|
|
||||||
|
|
||||||
// update linked records
|
|
||||||
await linkRecords.updateLinks({
|
|
||||||
instanceId,
|
|
||||||
eventType: linkRecords.EventType.MODEL_DELETE,
|
|
||||||
model: modelToDelete,
|
|
||||||
})
|
|
||||||
|
|
||||||
// don't remove the table itself until very end
|
|
||||||
await db.remove(modelToDelete)
|
|
||||||
|
|
||||||
ctx.eventEmitter &&
|
|
||||||
ctx.eventEmitter.emitModel(`model:delete`, instanceId, modelToDelete)
|
|
||||||
ctx.status = 200
|
|
||||||
ctx.message = `Model ${ctx.params.modelId} deleted.`
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.validateCSVSchema = async function(ctx) {
|
|
||||||
const { file, schema = {} } = ctx.request.body
|
|
||||||
const result = await csvParser.parse(file.path, schema)
|
|
||||||
ctx.body = {
|
|
||||||
schema: result,
|
|
||||||
path: file.path,
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,382 +0,0 @@
|
||||||
const CouchDB = require("../../db")
|
|
||||||
const validateJs = require("validate.js")
|
|
||||||
const linkRecords = require("../../db/linkedRecords")
|
|
||||||
const {
|
|
||||||
getRecordParams,
|
|
||||||
generateRecordID,
|
|
||||||
DocumentTypes,
|
|
||||||
SEPARATOR,
|
|
||||||
} = require("../../db/utils")
|
|
||||||
const { cloneDeep } = require("lodash")
|
|
||||||
|
|
||||||
const MODEL_VIEW_BEGINS_WITH = `all${SEPARATOR}${DocumentTypes.MODEL}${SEPARATOR}`
|
|
||||||
|
|
||||||
validateJs.extend(validateJs.validators.datetime, {
|
|
||||||
parse: function(value) {
|
|
||||||
return new Date(value).getTime()
|
|
||||||
},
|
|
||||||
// Input is a unix timestamp
|
|
||||||
format: function(value) {
|
|
||||||
return new Date(value).toISOString()
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
exports.patch = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
let record = await db.get(ctx.params.id)
|
|
||||||
const model = await db.get(record.modelId)
|
|
||||||
const patchfields = ctx.request.body
|
|
||||||
record = coerceRecordValues(record, model)
|
|
||||||
|
|
||||||
for (let key of Object.keys(patchfields)) {
|
|
||||||
if (!model.schema[key]) continue
|
|
||||||
record[key] = patchfields[key]
|
|
||||||
}
|
|
||||||
|
|
||||||
const validateResult = await validate({
|
|
||||||
record,
|
|
||||||
model,
|
|
||||||
})
|
|
||||||
|
|
||||||
if (!validateResult.valid) {
|
|
||||||
ctx.status = 400
|
|
||||||
ctx.body = {
|
|
||||||
status: 400,
|
|
||||||
errors: validateResult.errors,
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// returned record is cleaned and prepared for writing to DB
|
|
||||||
record = await linkRecords.updateLinks({
|
|
||||||
instanceId,
|
|
||||||
eventType: linkRecords.EventType.RECORD_UPDATE,
|
|
||||||
record,
|
|
||||||
modelId: record.modelId,
|
|
||||||
model,
|
|
||||||
})
|
|
||||||
const response = await db.put(record)
|
|
||||||
record._rev = response.rev
|
|
||||||
record.type = "record"
|
|
||||||
|
|
||||||
ctx.eventEmitter &&
|
|
||||||
ctx.eventEmitter.emitRecord(`record:update`, instanceId, record, model)
|
|
||||||
ctx.body = record
|
|
||||||
ctx.status = 200
|
|
||||||
ctx.message = `${model.name} updated successfully.`
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.save = async function(ctx) {
|
|
||||||
if (ctx.request.body.type === "delete") {
|
|
||||||
await bulkDelete(ctx)
|
|
||||||
} else {
|
|
||||||
await saveRecord(ctx)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.fetchView = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
const { stats, group, field } = ctx.query
|
|
||||||
const viewName = ctx.params.viewName
|
|
||||||
|
|
||||||
// if this is a model view being looked for just transfer to that
|
|
||||||
if (viewName.indexOf(MODEL_VIEW_BEGINS_WITH) === 0) {
|
|
||||||
ctx.params.modelId = viewName.substring(4)
|
|
||||||
await exports.fetchModelRecords(ctx)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const response = await db.query(`database/${viewName}`, {
|
|
||||||
include_docs: !stats,
|
|
||||||
group,
|
|
||||||
})
|
|
||||||
|
|
||||||
if (stats) {
|
|
||||||
response.rows = response.rows.map(row => ({
|
|
||||||
group: row.key,
|
|
||||||
field,
|
|
||||||
...row.value,
|
|
||||||
avg: row.value.sum / row.value.count,
|
|
||||||
}))
|
|
||||||
} else {
|
|
||||||
response.rows = response.rows.map(row => row.doc)
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.body = await linkRecords.attachLinkInfo(instanceId, response.rows)
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.fetchModelRecords = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
const response = await db.allDocs(
|
|
||||||
getRecordParams(ctx.params.modelId, null, {
|
|
||||||
include_docs: true,
|
|
||||||
})
|
|
||||||
)
|
|
||||||
ctx.body = response.rows.map(row => row.doc)
|
|
||||||
ctx.body = await linkRecords.attachLinkInfo(
|
|
||||||
instanceId,
|
|
||||||
response.rows.map(row => row.doc)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.search = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
const response = await db.allDocs({
|
|
||||||
include_docs: true,
|
|
||||||
...ctx.request.body,
|
|
||||||
})
|
|
||||||
ctx.body = await linkRecords.attachLinkInfo(
|
|
||||||
instanceId,
|
|
||||||
response.rows.map(row => row.doc)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.find = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
const record = await db.get(ctx.params.recordId)
|
|
||||||
if (record.modelId !== ctx.params.modelId) {
|
|
||||||
ctx.throw(400, "Supplied modelId does not match the records modelId")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ctx.body = await linkRecords.attachLinkInfo(instanceId, record)
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.destroy = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
const record = await db.get(ctx.params.recordId)
|
|
||||||
if (record.modelId !== ctx.params.modelId) {
|
|
||||||
ctx.throw(400, "Supplied modelId doesn't match the record's modelId")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
await linkRecords.updateLinks({
|
|
||||||
instanceId,
|
|
||||||
eventType: linkRecords.EventType.RECORD_DELETE,
|
|
||||||
record,
|
|
||||||
modelId: record.modelId,
|
|
||||||
})
|
|
||||||
ctx.body = await db.remove(ctx.params.recordId, ctx.params.revId)
|
|
||||||
ctx.status = 200
|
|
||||||
|
|
||||||
// for automations include the record that was deleted
|
|
||||||
ctx.record = record
|
|
||||||
ctx.eventEmitter &&
|
|
||||||
ctx.eventEmitter.emitRecord(`record:delete`, instanceId, record)
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.validate = async function(ctx) {
|
|
||||||
const errors = await validate({
|
|
||||||
instanceId: ctx.user.instanceId,
|
|
||||||
modelId: ctx.params.modelId,
|
|
||||||
record: ctx.request.body,
|
|
||||||
})
|
|
||||||
ctx.status = 200
|
|
||||||
ctx.body = errors
|
|
||||||
}
|
|
||||||
|
|
||||||
async function validate({ instanceId, modelId, record, model }) {
|
|
||||||
if (!model) {
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
model = await db.get(modelId)
|
|
||||||
}
|
|
||||||
const errors = {}
|
|
||||||
for (let fieldName of Object.keys(model.schema)) {
|
|
||||||
const res = validateJs.single(
|
|
||||||
record[fieldName],
|
|
||||||
model.schema[fieldName].constraints
|
|
||||||
)
|
|
||||||
if (res) errors[fieldName] = res
|
|
||||||
}
|
|
||||||
return { valid: Object.keys(errors).length === 0, errors }
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.fetchEnrichedRecord = async function(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
const modelId = ctx.params.modelId
|
|
||||||
const recordId = ctx.params.recordId
|
|
||||||
if (instanceId == null || modelId == null || recordId == null) {
|
|
||||||
ctx.status = 400
|
|
||||||
ctx.body = {
|
|
||||||
status: 400,
|
|
||||||
error:
|
|
||||||
"Cannot handle request, URI params have not been successfully prepared.",
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// need model to work out where links go in record
|
|
||||||
const [model, record] = await Promise.all([db.get(modelId), db.get(recordId)])
|
|
||||||
// get the link docs
|
|
||||||
const linkVals = await linkRecords.getLinkDocuments({
|
|
||||||
instanceId,
|
|
||||||
modelId,
|
|
||||||
recordId,
|
|
||||||
})
|
|
||||||
// look up the actual records based on the ids
|
|
||||||
const response = await db.allDocs({
|
|
||||||
include_docs: true,
|
|
||||||
keys: linkVals.map(linkVal => linkVal.id),
|
|
||||||
})
|
|
||||||
// need to include the IDs in these records for any links they may have
|
|
||||||
let linkedRecords = await linkRecords.attachLinkInfo(
|
|
||||||
instanceId,
|
|
||||||
response.rows.map(row => row.doc)
|
|
||||||
)
|
|
||||||
// insert the link records in the correct place throughout the main record
|
|
||||||
for (let fieldName of Object.keys(model.schema)) {
|
|
||||||
let field = model.schema[fieldName]
|
|
||||||
if (field.type === "link") {
|
|
||||||
record[fieldName] = linkedRecords.filter(
|
|
||||||
linkRecord => linkRecord.modelId === field.modelId
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ctx.body = record
|
|
||||||
ctx.status = 200
|
|
||||||
}
|
|
||||||
|
|
||||||
function coerceRecordValues(rec, model) {
|
|
||||||
const record = cloneDeep(rec)
|
|
||||||
for (let [key, value] of Object.entries(record)) {
|
|
||||||
const field = model.schema[key]
|
|
||||||
if (!field) continue
|
|
||||||
|
|
||||||
// eslint-disable-next-line no-prototype-builtins
|
|
||||||
if (TYPE_TRANSFORM_MAP[field.type].hasOwnProperty(value)) {
|
|
||||||
record[key] = TYPE_TRANSFORM_MAP[field.type][value]
|
|
||||||
} else if (TYPE_TRANSFORM_MAP[field.type].parse) {
|
|
||||||
record[key] = TYPE_TRANSFORM_MAP[field.type].parse(value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return record
|
|
||||||
}
|
|
||||||
|
|
||||||
const TYPE_TRANSFORM_MAP = {
|
|
||||||
link: {
|
|
||||||
"": [],
|
|
||||||
[null]: [],
|
|
||||||
[undefined]: undefined,
|
|
||||||
},
|
|
||||||
options: {
|
|
||||||
"": "",
|
|
||||||
[null]: "",
|
|
||||||
[undefined]: undefined,
|
|
||||||
},
|
|
||||||
string: {
|
|
||||||
"": "",
|
|
||||||
[null]: "",
|
|
||||||
[undefined]: undefined,
|
|
||||||
},
|
|
||||||
number: {
|
|
||||||
"": null,
|
|
||||||
[null]: null,
|
|
||||||
[undefined]: undefined,
|
|
||||||
parse: n => parseFloat(n),
|
|
||||||
},
|
|
||||||
datetime: {
|
|
||||||
"": null,
|
|
||||||
[undefined]: undefined,
|
|
||||||
[null]: null,
|
|
||||||
},
|
|
||||||
attachment: {
|
|
||||||
"": [],
|
|
||||||
[null]: [],
|
|
||||||
[undefined]: undefined,
|
|
||||||
},
|
|
||||||
boolean: {
|
|
||||||
"": null,
|
|
||||||
[null]: null,
|
|
||||||
[undefined]: undefined,
|
|
||||||
true: true,
|
|
||||||
false: false,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
async function bulkDelete(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const { records } = ctx.request.body
|
|
||||||
const db = new CouchDB(ctx.user.instanceId)
|
|
||||||
|
|
||||||
await db.bulkDocs(
|
|
||||||
records.map(
|
|
||||||
record => ({ ...record, _deleted: true }),
|
|
||||||
err => {
|
|
||||||
if (err) {
|
|
||||||
ctx.status = 500
|
|
||||||
} else {
|
|
||||||
records.forEach(record => {
|
|
||||||
ctx.eventEmitter &&
|
|
||||||
ctx.eventEmitter.emitRecord(`record:delete`, instanceId, record)
|
|
||||||
})
|
|
||||||
ctx.status = 200
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
async function saveRecord(ctx) {
|
|
||||||
const instanceId = ctx.user.instanceId
|
|
||||||
const db = new CouchDB(instanceId)
|
|
||||||
let record = ctx.request.body
|
|
||||||
record.modelId = ctx.params.modelId
|
|
||||||
|
|
||||||
if (!record._rev && !record._id) {
|
|
||||||
record._id = generateRecordID(record.modelId)
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the record obj had an _id then it will have been retrieved
|
|
||||||
const existingRecord = ctx.preExisting
|
|
||||||
|
|
||||||
const model = await db.get(record.modelId)
|
|
||||||
|
|
||||||
record = coerceRecordValues(record, model)
|
|
||||||
|
|
||||||
const validateResult = await validate({
|
|
||||||
record,
|
|
||||||
model,
|
|
||||||
})
|
|
||||||
|
|
||||||
if (!validateResult.valid) {
|
|
||||||
ctx.status = 400
|
|
||||||
ctx.body = {
|
|
||||||
status: 400,
|
|
||||||
errors: validateResult.errors,
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// make sure link records are up to date
|
|
||||||
record = await linkRecords.updateLinks({
|
|
||||||
instanceId,
|
|
||||||
eventType: linkRecords.EventType.RECORD_SAVE,
|
|
||||||
record,
|
|
||||||
modelId: record.modelId,
|
|
||||||
model,
|
|
||||||
})
|
|
||||||
|
|
||||||
if (existingRecord) {
|
|
||||||
const response = await db.put(record)
|
|
||||||
record._rev = response.rev
|
|
||||||
record.type = "record"
|
|
||||||
ctx.body = record
|
|
||||||
ctx.status = 200
|
|
||||||
ctx.message = `${model.name} updated successfully.`
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
record.type = "record"
|
|
||||||
const response = await db.post(record)
|
|
||||||
record._rev = response.rev
|
|
||||||
|
|
||||||
ctx.eventEmitter &&
|
|
||||||
ctx.eventEmitter.emitRecord(`record:save`, instanceId, record, model)
|
|
||||||
ctx.body = record
|
|
||||||
ctx.status = 200
|
|
||||||
ctx.message = `${model.name} created successfully`
|
|
||||||
}
|
|
|
@ -0,0 +1,350 @@
|
||||||
|
const CouchDB = require("../../db")
|
||||||
|
const validateJs = require("validate.js")
|
||||||
|
const linkRows = require("../../db/linkedRows")
|
||||||
|
const {
|
||||||
|
getRowParams,
|
||||||
|
generateRowID,
|
||||||
|
DocumentTypes,
|
||||||
|
SEPARATOR,
|
||||||
|
} = require("../../db/utils")
|
||||||
|
const { cloneDeep } = require("lodash")
|
||||||
|
|
||||||
|
const TABLE_VIEW_BEGINS_WITH = `all${SEPARATOR}${DocumentTypes.TABLE}${SEPARATOR}`
|
||||||
|
|
||||||
|
validateJs.extend(validateJs.validators.datetime, {
|
||||||
|
parse: function(value) {
|
||||||
|
return new Date(value).getTime()
|
||||||
|
},
|
||||||
|
// Input is a unix timestamp
|
||||||
|
format: function(value) {
|
||||||
|
return new Date(value).toISOString()
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
exports.patch = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
let row = await db.get(ctx.params.id)
|
||||||
|
const table = await db.get(row.tableId)
|
||||||
|
const patchfields = ctx.request.body
|
||||||
|
row = coerceRowValues(row, table)
|
||||||
|
|
||||||
|
for (let key of Object.keys(patchfields)) {
|
||||||
|
if (!table.schema[key]) continue
|
||||||
|
row[key] = patchfields[key]
|
||||||
|
}
|
||||||
|
|
||||||
|
const validateResult = await validate({
|
||||||
|
row,
|
||||||
|
table,
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!validateResult.valid) {
|
||||||
|
ctx.status = 400
|
||||||
|
ctx.body = {
|
||||||
|
status: 400,
|
||||||
|
errors: validateResult.errors,
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// returned row is cleaned and prepared for writing to DB
|
||||||
|
row = await linkRows.updateLinks({
|
||||||
|
instanceId,
|
||||||
|
eventType: linkRows.EventType.ROW_UPDATE,
|
||||||
|
row,
|
||||||
|
tableId: row.tableId,
|
||||||
|
table,
|
||||||
|
})
|
||||||
|
const response = await db.put(row)
|
||||||
|
row._rev = response.rev
|
||||||
|
row.type = "row"
|
||||||
|
|
||||||
|
ctx.eventEmitter &&
|
||||||
|
ctx.eventEmitter.emitRow(`row:update`, instanceId, row, table)
|
||||||
|
ctx.body = row
|
||||||
|
ctx.status = 200
|
||||||
|
ctx.message = `${table.name} updated successfully.`
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.save = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
let row = ctx.request.body
|
||||||
|
row.tableId = ctx.params.tableId
|
||||||
|
|
||||||
|
if (!row._rev && !row._id) {
|
||||||
|
row._id = generateRowID(row.tableId)
|
||||||
|
}
|
||||||
|
|
||||||
|
// if the row obj had an _id then it will have been retrieved
|
||||||
|
const existingRow = ctx.preExisting
|
||||||
|
|
||||||
|
const table = await db.get(row.tableId)
|
||||||
|
|
||||||
|
row = coerceRowValues(row, table)
|
||||||
|
|
||||||
|
const validateResult = await validate({
|
||||||
|
row,
|
||||||
|
table,
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!validateResult.valid) {
|
||||||
|
ctx.status = 400
|
||||||
|
ctx.body = {
|
||||||
|
status: 400,
|
||||||
|
errors: validateResult.errors,
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// make sure link rows are up to date
|
||||||
|
row = await linkRows.updateLinks({
|
||||||
|
instanceId,
|
||||||
|
eventType: linkRows.EventType.ROW_SAVE,
|
||||||
|
row,
|
||||||
|
tableId: row.tableId,
|
||||||
|
table,
|
||||||
|
})
|
||||||
|
|
||||||
|
if (existingRow) {
|
||||||
|
const response = await db.put(row)
|
||||||
|
row._rev = response.rev
|
||||||
|
row.type = "row"
|
||||||
|
ctx.body = row
|
||||||
|
ctx.status = 200
|
||||||
|
ctx.message = `${table.name} updated successfully.`
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
row.type = "row"
|
||||||
|
const response = await db.post(row)
|
||||||
|
row._rev = response.rev
|
||||||
|
|
||||||
|
ctx.eventEmitter &&
|
||||||
|
ctx.eventEmitter.emitRow(`row:save`, instanceId, row, table)
|
||||||
|
ctx.body = row
|
||||||
|
ctx.status = 200
|
||||||
|
ctx.message = `${table.name} created successfully`
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.fetchView = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
const { stats, group, field } = ctx.query
|
||||||
|
const viewName = ctx.params.viewName
|
||||||
|
|
||||||
|
// if this is a table view being looked for just transfer to that
|
||||||
|
if (viewName.indexOf(TABLE_VIEW_BEGINS_WITH) === 0) {
|
||||||
|
ctx.params.tableId = viewName.substring(4)
|
||||||
|
await exports.fetchTableRows(ctx)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await db.query(`database/${viewName}`, {
|
||||||
|
include_docs: !stats,
|
||||||
|
group,
|
||||||
|
})
|
||||||
|
|
||||||
|
if (stats) {
|
||||||
|
response.rows = response.rows.map(row => ({
|
||||||
|
group: row.key,
|
||||||
|
field,
|
||||||
|
...row.value,
|
||||||
|
avg: row.value.sum / row.value.count,
|
||||||
|
}))
|
||||||
|
} else {
|
||||||
|
response.rows = response.rows.map(row => row.doc)
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx.body = await linkRows.attachLinkInfo(instanceId, response.rows)
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.fetchTableRows = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
const response = await db.allDocs(
|
||||||
|
getRowParams(ctx.params.tableId, null, {
|
||||||
|
include_docs: true,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
ctx.body = response.rows.map(row => row.doc)
|
||||||
|
ctx.body = await linkRows.attachLinkInfo(
|
||||||
|
instanceId,
|
||||||
|
response.rows.map(row => row.doc)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.search = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
const response = await db.allDocs({
|
||||||
|
include_docs: true,
|
||||||
|
...ctx.request.body,
|
||||||
|
})
|
||||||
|
ctx.body = await linkRows.attachLinkInfo(
|
||||||
|
instanceId,
|
||||||
|
response.rows.map(row => row.doc)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.find = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
const row = await db.get(ctx.params.rowId)
|
||||||
|
if (row.tableId !== ctx.params.tableId) {
|
||||||
|
ctx.throw(400, "Supplied tableId does not match the rows tableId")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.body = await linkRows.attachLinkInfo(instanceId, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.destroy = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
const row = await db.get(ctx.params.rowId)
|
||||||
|
if (row.tableId !== ctx.params.tableId) {
|
||||||
|
ctx.throw(400, "Supplied tableId doesn't match the row's tableId")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
await linkRows.updateLinks({
|
||||||
|
instanceId,
|
||||||
|
eventType: linkRows.EventType.ROW_DELETE,
|
||||||
|
row,
|
||||||
|
tableId: row.tableId,
|
||||||
|
})
|
||||||
|
ctx.body = await db.remove(ctx.params.rowId, ctx.params.revId)
|
||||||
|
ctx.status = 200
|
||||||
|
|
||||||
|
// for automations include the row that was deleted
|
||||||
|
ctx.row = row
|
||||||
|
ctx.eventEmitter && ctx.eventEmitter.emitRow(`row:delete`, instanceId, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.validate = async function(ctx) {
|
||||||
|
const errors = await validate({
|
||||||
|
instanceId: ctx.user.instanceId,
|
||||||
|
tableId: ctx.params.tableId,
|
||||||
|
row: ctx.request.body,
|
||||||
|
})
|
||||||
|
ctx.status = 200
|
||||||
|
ctx.body = errors
|
||||||
|
}
|
||||||
|
|
||||||
|
async function validate({ instanceId, tableId, row, table }) {
|
||||||
|
if (!table) {
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
table = await db.get(tableId)
|
||||||
|
}
|
||||||
|
const errors = {}
|
||||||
|
for (let fieldName of Object.keys(table.schema)) {
|
||||||
|
const res = validateJs.single(
|
||||||
|
row[fieldName],
|
||||||
|
table.schema[fieldName].constraints
|
||||||
|
)
|
||||||
|
if (res) errors[fieldName] = res
|
||||||
|
}
|
||||||
|
return { valid: Object.keys(errors).length === 0, errors }
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.fetchEnrichedRow = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
const tableId = ctx.params.tableId
|
||||||
|
const rowId = ctx.params.rowId
|
||||||
|
if (instanceId == null || tableId == null || rowId == null) {
|
||||||
|
ctx.status = 400
|
||||||
|
ctx.body = {
|
||||||
|
status: 400,
|
||||||
|
error:
|
||||||
|
"Cannot handle request, URI params have not been successfully prepared.",
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// need table to work out where links go in row
|
||||||
|
const [table, row] = await Promise.all([db.get(tableId), db.get(rowId)])
|
||||||
|
// get the link docs
|
||||||
|
const linkVals = await linkRows.getLinkDocuments({
|
||||||
|
instanceId,
|
||||||
|
tableId,
|
||||||
|
rowId,
|
||||||
|
})
|
||||||
|
// look up the actual rows based on the ids
|
||||||
|
const response = await db.allDocs({
|
||||||
|
include_docs: true,
|
||||||
|
keys: linkVals.map(linkVal => linkVal.id),
|
||||||
|
})
|
||||||
|
// need to include the IDs in these rows for any links they may have
|
||||||
|
let linkedRows = await linkRows.attachLinkInfo(
|
||||||
|
instanceId,
|
||||||
|
response.rows.map(row => row.doc)
|
||||||
|
)
|
||||||
|
// insert the link rows in the correct place throughout the main row
|
||||||
|
for (let fieldName of Object.keys(table.schema)) {
|
||||||
|
let field = table.schema[fieldName]
|
||||||
|
if (field.type === "link") {
|
||||||
|
row[fieldName] = linkedRows.filter(
|
||||||
|
linkRow => linkRow.tableId === field.tableId
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ctx.body = row
|
||||||
|
ctx.status = 200
|
||||||
|
}
|
||||||
|
|
||||||
|
function coerceRowValues(rec, table) {
|
||||||
|
const row = cloneDeep(rec)
|
||||||
|
for (let [key, value] of Object.entries(row)) {
|
||||||
|
const field = table.schema[key]
|
||||||
|
if (!field) continue
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-prototype-builtins
|
||||||
|
if (TYPE_TRANSFORM_MAP[field.type].hasOwnProperty(value)) {
|
||||||
|
row[key] = TYPE_TRANSFORM_MAP[field.type][value]
|
||||||
|
} else if (TYPE_TRANSFORM_MAP[field.type].parse) {
|
||||||
|
row[key] = TYPE_TRANSFORM_MAP[field.type].parse(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return row
|
||||||
|
}
|
||||||
|
|
||||||
|
const TYPE_TRANSFORM_MAP = {
|
||||||
|
link: {
|
||||||
|
"": [],
|
||||||
|
[null]: [],
|
||||||
|
[undefined]: undefined,
|
||||||
|
},
|
||||||
|
options: {
|
||||||
|
"": "",
|
||||||
|
[null]: "",
|
||||||
|
[undefined]: undefined,
|
||||||
|
},
|
||||||
|
string: {
|
||||||
|
"": "",
|
||||||
|
[null]: "",
|
||||||
|
[undefined]: undefined,
|
||||||
|
},
|
||||||
|
number: {
|
||||||
|
"": null,
|
||||||
|
[null]: null,
|
||||||
|
[undefined]: undefined,
|
||||||
|
parse: n => parseFloat(n),
|
||||||
|
},
|
||||||
|
datetime: {
|
||||||
|
"": null,
|
||||||
|
[undefined]: undefined,
|
||||||
|
[null]: null,
|
||||||
|
},
|
||||||
|
attachment: {
|
||||||
|
"": [],
|
||||||
|
[null]: [],
|
||||||
|
[undefined]: undefined,
|
||||||
|
},
|
||||||
|
boolean: {
|
||||||
|
"": null,
|
||||||
|
[null]: null,
|
||||||
|
[undefined]: undefined,
|
||||||
|
true: true,
|
||||||
|
false: false,
|
||||||
|
},
|
||||||
|
}
|
|
@ -0,0 +1,144 @@
|
||||||
|
const CouchDB = require("../../db")
|
||||||
|
const linkRows = require("../../db/linkedRows")
|
||||||
|
const csvParser = require("../../utilities/csvParser")
|
||||||
|
const {
|
||||||
|
getRowParams,
|
||||||
|
getTableParams,
|
||||||
|
generateTableID,
|
||||||
|
generateRowID,
|
||||||
|
} = require("../../db/utils")
|
||||||
|
|
||||||
|
exports.fetch = async function(ctx) {
|
||||||
|
const db = new CouchDB(ctx.user.instanceId)
|
||||||
|
const body = await db.allDocs(
|
||||||
|
getTableParams(null, {
|
||||||
|
include_docs: true,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
ctx.body = body.rows.map(row => row.doc)
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.find = async function(ctx) {
|
||||||
|
const db = new CouchDB(ctx.user.instanceId)
|
||||||
|
ctx.body = await db.get(ctx.params.id)
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.save = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
const { dataImport, ...rest } = ctx.request.body
|
||||||
|
const tableToSave = {
|
||||||
|
type: "table",
|
||||||
|
_id: generateTableID(),
|
||||||
|
views: {},
|
||||||
|
...rest,
|
||||||
|
}
|
||||||
|
let renameDocs = []
|
||||||
|
|
||||||
|
// if the table obj had an _id then it will have been retrieved
|
||||||
|
const oldTable = ctx.preExisting
|
||||||
|
|
||||||
|
// rename row fields when table column is renamed
|
||||||
|
const { _rename } = tableToSave
|
||||||
|
if (_rename && tableToSave.schema[_rename.updated].type === "link") {
|
||||||
|
throw "Cannot rename a linked field."
|
||||||
|
} else if (_rename && tableToSave.primaryDisplay === _rename.old) {
|
||||||
|
throw "Cannot rename the primary display field."
|
||||||
|
} else if (_rename) {
|
||||||
|
const rows = await db.allDocs(
|
||||||
|
getRowParams(tableToSave._id, null, {
|
||||||
|
include_docs: true,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
renameDocs = rows.rows.map(({ doc }) => {
|
||||||
|
doc[_rename.updated] = doc[_rename.old]
|
||||||
|
delete doc[_rename.old]
|
||||||
|
return doc
|
||||||
|
})
|
||||||
|
delete tableToSave._rename
|
||||||
|
}
|
||||||
|
|
||||||
|
// update schema of non-statistics views when new columns are added
|
||||||
|
for (let view in tableToSave.views) {
|
||||||
|
const tableView = tableToSave.views[view]
|
||||||
|
if (!tableView) continue
|
||||||
|
|
||||||
|
if (tableView.schema.group || tableView.schema.field) continue
|
||||||
|
tableView.schema = tableToSave.schema
|
||||||
|
}
|
||||||
|
|
||||||
|
// update linked rows
|
||||||
|
await linkRows.updateLinks({
|
||||||
|
instanceId,
|
||||||
|
eventType: oldTable
|
||||||
|
? linkRows.EventType.TABLE_UPDATED
|
||||||
|
: linkRows.EventType.TABLE_SAVE,
|
||||||
|
table: tableToSave,
|
||||||
|
oldTable: oldTable,
|
||||||
|
})
|
||||||
|
|
||||||
|
// don't perform any updates until relationships have been
|
||||||
|
// checked by the updateLinks function
|
||||||
|
if (renameDocs.length !== 0) {
|
||||||
|
await db.bulkDocs(renameDocs)
|
||||||
|
}
|
||||||
|
const result = await db.post(tableToSave)
|
||||||
|
tableToSave._rev = result.rev
|
||||||
|
|
||||||
|
ctx.eventEmitter &&
|
||||||
|
ctx.eventEmitter.emitTable(`table:save`, instanceId, tableToSave)
|
||||||
|
|
||||||
|
if (dataImport && dataImport.path) {
|
||||||
|
// Populate the table with rows imported from CSV in a bulk update
|
||||||
|
const data = await csvParser.transform(dataImport)
|
||||||
|
|
||||||
|
for (let row of data) {
|
||||||
|
row._id = generateRowID(tableToSave._id)
|
||||||
|
row.tableId = tableToSave._id
|
||||||
|
}
|
||||||
|
|
||||||
|
await db.bulkDocs(data)
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx.status = 200
|
||||||
|
ctx.message = `Table ${ctx.request.body.name} saved successfully.`
|
||||||
|
ctx.body = tableToSave
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.destroy = async function(ctx) {
|
||||||
|
const instanceId = ctx.user.instanceId
|
||||||
|
const db = new CouchDB(instanceId)
|
||||||
|
const tableToDelete = await db.get(ctx.params.tableId)
|
||||||
|
|
||||||
|
// Delete all rows for that table
|
||||||
|
const rows = await db.allDocs(
|
||||||
|
getRowParams(ctx.params.tableId, null, {
|
||||||
|
include_docs: true,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
await db.bulkDocs(rows.rows.map(row => ({ ...row.doc, _deleted: true })))
|
||||||
|
|
||||||
|
// update linked rows
|
||||||
|
await linkRows.updateLinks({
|
||||||
|
instanceId,
|
||||||
|
eventType: linkRows.EventType.TABLE_DELETE,
|
||||||
|
table: tableToDelete,
|
||||||
|
})
|
||||||
|
|
||||||
|
// don't remove the table itself until very end
|
||||||
|
await db.remove(tableToDelete)
|
||||||
|
|
||||||
|
ctx.eventEmitter &&
|
||||||
|
ctx.eventEmitter.emitTable(`table:delete`, instanceId, tableToDelete)
|
||||||
|
ctx.status = 200
|
||||||
|
ctx.message = `Table ${ctx.params.tableId} deleted.`
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.validateCSVSchema = async function(ctx) {
|
||||||
|
const { file, schema = {} } = ctx.request.body
|
||||||
|
const result = await csvParser.parse(file.path, schema)
|
||||||
|
ctx.body = {
|
||||||
|
schema: result,
|
||||||
|
path: file.path,
|
||||||
|
}
|
||||||
|
}
|
|
@ -4,7 +4,7 @@ const fs = require("fs")
|
||||||
const { join } = require("../../../utilities/centralPath")
|
const { join } = require("../../../utilities/centralPath")
|
||||||
const os = require("os")
|
const os = require("os")
|
||||||
const exporters = require("./exporters")
|
const exporters = require("./exporters")
|
||||||
const { fetchView } = require("../record")
|
const { fetchView } = require("../row")
|
||||||
|
|
||||||
const controller = {
|
const controller = {
|
||||||
fetch: async ctx => {
|
fetch: async ctx => {
|
||||||
|
@ -45,21 +45,21 @@ const controller = {
|
||||||
|
|
||||||
await db.put(designDoc)
|
await db.put(designDoc)
|
||||||
|
|
||||||
// add views to model document
|
// add views to table document
|
||||||
const model = await db.get(ctx.request.body.modelId)
|
const table = await db.get(ctx.request.body.tableId)
|
||||||
if (!model.views) model.views = {}
|
if (!table.views) table.views = {}
|
||||||
if (!view.meta.schema) {
|
if (!view.meta.schema) {
|
||||||
view.meta.schema = model.schema
|
view.meta.schema = table.schema
|
||||||
}
|
}
|
||||||
model.views[viewToSave.name] = view.meta
|
table.views[viewToSave.name] = view.meta
|
||||||
|
|
||||||
if (originalName) {
|
if (originalName) {
|
||||||
delete model.views[originalName]
|
delete table.views[originalName]
|
||||||
}
|
}
|
||||||
|
|
||||||
await db.put(model)
|
await db.put(table)
|
||||||
|
|
||||||
ctx.body = model.views[viewToSave.name]
|
ctx.body = table.views[viewToSave.name]
|
||||||
ctx.message = `View ${viewToSave.name} saved successfully.`
|
ctx.message = `View ${viewToSave.name} saved successfully.`
|
||||||
},
|
},
|
||||||
destroy: async ctx => {
|
destroy: async ctx => {
|
||||||
|
@ -74,9 +74,9 @@ const controller = {
|
||||||
|
|
||||||
await db.put(designDoc)
|
await db.put(designDoc)
|
||||||
|
|
||||||
const model = await db.get(view.meta.modelId)
|
const table = await db.get(view.meta.tableId)
|
||||||
delete model.views[viewName]
|
delete table.views[viewName]
|
||||||
await db.put(model)
|
await db.put(table)
|
||||||
|
|
||||||
ctx.body = view
|
ctx.body = view
|
||||||
ctx.message = `View ${ctx.params.viewName} saved successfully.`
|
ctx.message = `View ${ctx.params.viewName} saved successfully.`
|
||||||
|
@ -85,7 +85,7 @@ const controller = {
|
||||||
const view = ctx.request.body
|
const view = ctx.request.body
|
||||||
const format = ctx.query.format
|
const format = ctx.query.format
|
||||||
|
|
||||||
// Fetch view records
|
// Fetch view rows
|
||||||
ctx.params.viewName = view.name
|
ctx.params.viewName = view.name
|
||||||
ctx.query.group = view.groupBy
|
ctx.query.group = view.groupBy
|
||||||
if (view.field) {
|
if (view.field) {
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
exports[`viewBuilder Calculate creates a view with the calculation statistics schema 1`] = `
|
exports[`viewBuilder Calculate creates a view with the calculation statistics schema 1`] = `
|
||||||
Object {
|
Object {
|
||||||
"map": "function (doc) {
|
"map": "function (doc) {
|
||||||
if (doc.modelId === \\"14f1c4e94d6a47b682ce89d35d4c78b0\\" ) {
|
if (doc.tableId === \\"14f1c4e94d6a47b682ce89d35d4c78b0\\" ) {
|
||||||
emit(doc[\\"_id\\"], doc[\\"myField\\"]);
|
emit(doc[\\"_id\\"], doc[\\"myField\\"]);
|
||||||
}
|
}
|
||||||
}",
|
}",
|
||||||
|
@ -12,7 +12,6 @@ Object {
|
||||||
"field": "myField",
|
"field": "myField",
|
||||||
"filters": Array [],
|
"filters": Array [],
|
||||||
"groupBy": undefined,
|
"groupBy": undefined,
|
||||||
"modelId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
|
||||||
"schema": Object {
|
"schema": Object {
|
||||||
"avg": Object {
|
"avg": Object {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
|
@ -36,6 +35,7 @@ Object {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"tableId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
||||||
},
|
},
|
||||||
"reduce": "_stats",
|
"reduce": "_stats",
|
||||||
}
|
}
|
||||||
|
@ -44,7 +44,7 @@ Object {
|
||||||
exports[`viewBuilder Filter creates a view with multiple filters and conjunctions 1`] = `
|
exports[`viewBuilder Filter creates a view with multiple filters and conjunctions 1`] = `
|
||||||
Object {
|
Object {
|
||||||
"map": "function (doc) {
|
"map": "function (doc) {
|
||||||
if (doc.modelId === \\"14f1c4e94d6a47b682ce89d35d4c78b0\\" && doc[\\"Name\\"] === \\"Test\\" || doc[\\"Yes\\"] > \\"Value\\") {
|
if (doc.tableId === \\"14f1c4e94d6a47b682ce89d35d4c78b0\\" && doc[\\"Name\\"] === \\"Test\\" || doc[\\"Yes\\"] > \\"Value\\") {
|
||||||
emit(doc._id);
|
emit(doc._id);
|
||||||
}
|
}
|
||||||
}",
|
}",
|
||||||
|
@ -65,8 +65,8 @@ Object {
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
"groupBy": undefined,
|
"groupBy": undefined,
|
||||||
"modelId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
|
||||||
"schema": null,
|
"schema": null,
|
||||||
|
"tableId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
@ -74,7 +74,7 @@ Object {
|
||||||
exports[`viewBuilder Group By creates a view emitting the group by field 1`] = `
|
exports[`viewBuilder Group By creates a view emitting the group by field 1`] = `
|
||||||
Object {
|
Object {
|
||||||
"map": "function (doc) {
|
"map": "function (doc) {
|
||||||
if (doc.modelId === \\"14f1c4e94d6a47b682ce89d35d4c78b0\\" ) {
|
if (doc.tableId === \\"14f1c4e94d6a47b682ce89d35d4c78b0\\" ) {
|
||||||
emit(doc[\\"age\\"], doc[\\"score\\"]);
|
emit(doc[\\"age\\"], doc[\\"score\\"]);
|
||||||
}
|
}
|
||||||
}",
|
}",
|
||||||
|
@ -83,8 +83,8 @@ Object {
|
||||||
"field": "score",
|
"field": "score",
|
||||||
"filters": Array [],
|
"filters": Array [],
|
||||||
"groupBy": "age",
|
"groupBy": "age",
|
||||||
"modelId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
|
||||||
"schema": null,
|
"schema": null,
|
||||||
|
"tableId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
||||||
},
|
},
|
||||||
"reduce": "_stats",
|
"reduce": "_stats",
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@ describe("viewBuilder", () => {
|
||||||
it("creates a view with multiple filters and conjunctions", () => {
|
it("creates a view with multiple filters and conjunctions", () => {
|
||||||
expect(viewTemplate({
|
expect(viewTemplate({
|
||||||
"name": "Test View",
|
"name": "Test View",
|
||||||
"modelId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
"tableId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
||||||
"filters": [{
|
"filters": [{
|
||||||
"value": "Test",
|
"value": "Test",
|
||||||
"condition": "EQUALS",
|
"condition": "EQUALS",
|
||||||
|
@ -27,7 +27,7 @@ describe("viewBuilder", () => {
|
||||||
"name": "Calculate View",
|
"name": "Calculate View",
|
||||||
"field": "myField",
|
"field": "myField",
|
||||||
"calculation": "stats",
|
"calculation": "stats",
|
||||||
"modelId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
"tableId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
||||||
"filters": []
|
"filters": []
|
||||||
})).toMatchSnapshot()
|
})).toMatchSnapshot()
|
||||||
})
|
})
|
||||||
|
@ -37,7 +37,7 @@ describe("viewBuilder", () => {
|
||||||
it("creates a view emitting the group by field", () => {
|
it("creates a view emitting the group by field", () => {
|
||||||
expect(viewTemplate({
|
expect(viewTemplate({
|
||||||
"name": "Test Scores Grouped By Age",
|
"name": "Test Scores Grouped By Age",
|
||||||
"modelId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
"tableId": "14f1c4e94d6a47b682ce89d35d4c78b0",
|
||||||
"groupBy": "age",
|
"groupBy": "age",
|
||||||
"field": "score",
|
"field": "score",
|
||||||
"filters": [],
|
"filters": [],
|
||||||
|
|
|
@ -90,12 +90,12 @@ function parseEmitExpression(field, groupBy) {
|
||||||
*
|
*
|
||||||
* @param {Object} viewDefinition - the JSON definition for a custom view.
|
* @param {Object} viewDefinition - the JSON definition for a custom view.
|
||||||
* field: field that calculations will be performed on
|
* field: field that calculations will be performed on
|
||||||
* modelId: modelId of the model this view was created from
|
* tableId: tableId of the table this view was created from
|
||||||
* groupBy: field that calculations will be grouped by. Field must be present for this to be useful
|
* groupBy: field that calculations will be grouped by. Field must be present for this to be useful
|
||||||
* filters: Array of filter objects containing predicates that are parsed into a JS expression
|
* filters: Array of filter objects containing predicates that are parsed into a JS expression
|
||||||
* calculation: an optional calculation to be performed over the view data.
|
* calculation: an optional calculation to be performed over the view data.
|
||||||
*/
|
*/
|
||||||
function viewTemplate({ field, modelId, groupBy, filters = [], calculation }) {
|
function viewTemplate({ field, tableId, groupBy, filters = [], calculation }) {
|
||||||
const parsedFilters = parseFilterExpression(filters)
|
const parsedFilters = parseFilterExpression(filters)
|
||||||
const filterExpression = parsedFilters ? `&& ${parsedFilters}` : ""
|
const filterExpression = parsedFilters ? `&& ${parsedFilters}` : ""
|
||||||
|
|
||||||
|
@ -115,14 +115,14 @@ function viewTemplate({ field, modelId, groupBy, filters = [], calculation }) {
|
||||||
return {
|
return {
|
||||||
meta: {
|
meta: {
|
||||||
field,
|
field,
|
||||||
modelId,
|
tableId,
|
||||||
groupBy,
|
groupBy,
|
||||||
filters,
|
filters,
|
||||||
schema,
|
schema,
|
||||||
calculation,
|
calculation,
|
||||||
},
|
},
|
||||||
map: `function (doc) {
|
map: `function (doc) {
|
||||||
if (doc.modelId === "${modelId}" ${filterExpression}) {
|
if (doc.tableId === "${tableId}" ${filterExpression}) {
|
||||||
${emitExpression}
|
${emitExpression}
|
||||||
}
|
}
|
||||||
}`,
|
}`,
|
||||||
|
|
|
@ -11,8 +11,8 @@ const {
|
||||||
instanceRoutes,
|
instanceRoutes,
|
||||||
clientRoutes,
|
clientRoutes,
|
||||||
applicationRoutes,
|
applicationRoutes,
|
||||||
recordRoutes,
|
rowRoutes,
|
||||||
modelRoutes,
|
tableRoutes,
|
||||||
viewRoutes,
|
viewRoutes,
|
||||||
staticRoutes,
|
staticRoutes,
|
||||||
componentRoutes,
|
componentRoutes,
|
||||||
|
@ -74,11 +74,11 @@ router.use(authRoutes.allowedMethods())
|
||||||
router.use(viewRoutes.routes())
|
router.use(viewRoutes.routes())
|
||||||
router.use(viewRoutes.allowedMethods())
|
router.use(viewRoutes.allowedMethods())
|
||||||
|
|
||||||
router.use(modelRoutes.routes())
|
router.use(tableRoutes.routes())
|
||||||
router.use(modelRoutes.allowedMethods())
|
router.use(tableRoutes.allowedMethods())
|
||||||
|
|
||||||
router.use(recordRoutes.routes())
|
router.use(rowRoutes.routes())
|
||||||
router.use(recordRoutes.allowedMethods())
|
router.use(rowRoutes.allowedMethods())
|
||||||
|
|
||||||
router.use(userRoutes.routes())
|
router.use(userRoutes.routes())
|
||||||
router.use(userRoutes.allowedMethods())
|
router.use(userRoutes.allowedMethods())
|
||||||
|
|
|
@ -4,8 +4,8 @@ const userRoutes = require("./user")
|
||||||
const instanceRoutes = require("./instance")
|
const instanceRoutes = require("./instance")
|
||||||
const clientRoutes = require("./client")
|
const clientRoutes = require("./client")
|
||||||
const applicationRoutes = require("./application")
|
const applicationRoutes = require("./application")
|
||||||
const modelRoutes = require("./model")
|
const tableRoutes = require("./table")
|
||||||
const recordRoutes = require("./record")
|
const rowRoutes = require("./row")
|
||||||
const viewRoutes = require("./view")
|
const viewRoutes = require("./view")
|
||||||
const staticRoutes = require("./static")
|
const staticRoutes = require("./static")
|
||||||
const componentRoutes = require("./component")
|
const componentRoutes = require("./component")
|
||||||
|
@ -24,8 +24,8 @@ module.exports = {
|
||||||
instanceRoutes,
|
instanceRoutes,
|
||||||
clientRoutes,
|
clientRoutes,
|
||||||
applicationRoutes,
|
applicationRoutes,
|
||||||
recordRoutes,
|
rowRoutes,
|
||||||
modelRoutes,
|
tableRoutes,
|
||||||
viewRoutes,
|
viewRoutes,
|
||||||
staticRoutes,
|
staticRoutes,
|
||||||
componentRoutes,
|
componentRoutes,
|
||||||
|
|
|
@ -1,27 +0,0 @@
|
||||||
const Router = require("@koa/router")
|
|
||||||
const modelController = require("../controllers/model")
|
|
||||||
const authorized = require("../../middleware/authorized")
|
|
||||||
const { BUILDER, READ_MODEL } = require("../../utilities/accessLevels")
|
|
||||||
|
|
||||||
const router = Router()
|
|
||||||
|
|
||||||
router
|
|
||||||
.get("/api/models", authorized(BUILDER), modelController.fetch)
|
|
||||||
.get(
|
|
||||||
"/api/models/:id",
|
|
||||||
authorized(READ_MODEL, ctx => ctx.params.id),
|
|
||||||
modelController.find
|
|
||||||
)
|
|
||||||
.post("/api/models", authorized(BUILDER), modelController.save)
|
|
||||||
.post(
|
|
||||||
"/api/models/csv/validate",
|
|
||||||
authorized(BUILDER),
|
|
||||||
modelController.validateCSVSchema
|
|
||||||
)
|
|
||||||
.delete(
|
|
||||||
"/api/models/:modelId/:revId",
|
|
||||||
authorized(BUILDER),
|
|
||||||
modelController.destroy
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = router
|
|
|
@ -1,49 +0,0 @@
|
||||||
const Router = require("@koa/router")
|
|
||||||
const recordController = require("../controllers/record")
|
|
||||||
const authorized = require("../../middleware/authorized")
|
|
||||||
const usage = require("../../middleware/usageQuota")
|
|
||||||
const { READ_MODEL, WRITE_MODEL } = require("../../utilities/accessLevels")
|
|
||||||
|
|
||||||
const router = Router()
|
|
||||||
|
|
||||||
router
|
|
||||||
.get(
|
|
||||||
"/api/:modelId/:recordId/enrich",
|
|
||||||
authorized(READ_MODEL, ctx => ctx.params.modelId),
|
|
||||||
recordController.fetchEnrichedRecord
|
|
||||||
)
|
|
||||||
.get(
|
|
||||||
"/api/:modelId/records",
|
|
||||||
authorized(READ_MODEL, ctx => ctx.params.modelId),
|
|
||||||
recordController.fetchModelRecords
|
|
||||||
)
|
|
||||||
.get(
|
|
||||||
"/api/:modelId/records/:recordId",
|
|
||||||
authorized(READ_MODEL, ctx => ctx.params.modelId),
|
|
||||||
recordController.find
|
|
||||||
)
|
|
||||||
.post("/api/records/search", recordController.search)
|
|
||||||
.post(
|
|
||||||
"/api/:modelId/records",
|
|
||||||
authorized(WRITE_MODEL, ctx => ctx.params.modelId),
|
|
||||||
usage,
|
|
||||||
recordController.save
|
|
||||||
)
|
|
||||||
.patch(
|
|
||||||
"/api/:modelId/records/:id",
|
|
||||||
authorized(WRITE_MODEL, ctx => ctx.params.modelId),
|
|
||||||
recordController.patch
|
|
||||||
)
|
|
||||||
.post(
|
|
||||||
"/api/:modelId/records/validate",
|
|
||||||
authorized(WRITE_MODEL, ctx => ctx.params.modelId),
|
|
||||||
recordController.validate
|
|
||||||
)
|
|
||||||
.delete(
|
|
||||||
"/api/:modelId/records/:recordId/:revId",
|
|
||||||
authorized(WRITE_MODEL, ctx => ctx.params.modelId),
|
|
||||||
usage,
|
|
||||||
recordController.destroy
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = router
|
|
|
@ -0,0 +1,49 @@
|
||||||
|
const Router = require("@koa/router")
|
||||||
|
const rowController = require("../controllers/row")
|
||||||
|
const authorized = require("../../middleware/authorized")
|
||||||
|
const usage = require("../../middleware/usageQuota")
|
||||||
|
const { READ_TABLE, WRITE_TABLE } = require("../../utilities/accessLevels")
|
||||||
|
|
||||||
|
const router = Router()
|
||||||
|
|
||||||
|
router
|
||||||
|
.get(
|
||||||
|
"/api/:tableId/:rowId/enrich",
|
||||||
|
authorized(READ_TABLE, ctx => ctx.params.tableId),
|
||||||
|
rowController.fetchEnrichedRow
|
||||||
|
)
|
||||||
|
.get(
|
||||||
|
"/api/:tableId/rows",
|
||||||
|
authorized(READ_TABLE, ctx => ctx.params.tableId),
|
||||||
|
rowController.fetchTableRows
|
||||||
|
)
|
||||||
|
.get(
|
||||||
|
"/api/:tableId/rows/:rowId",
|
||||||
|
authorized(READ_TABLE, ctx => ctx.params.tableId),
|
||||||
|
rowController.find
|
||||||
|
)
|
||||||
|
.post("/api/rows/search", rowController.search)
|
||||||
|
.post(
|
||||||
|
"/api/:tableId/rows",
|
||||||
|
authorized(WRITE_TABLE, ctx => ctx.params.tableId),
|
||||||
|
usage,
|
||||||
|
rowController.save
|
||||||
|
)
|
||||||
|
.patch(
|
||||||
|
"/api/:tableId/rows/:id",
|
||||||
|
authorized(WRITE_TABLE, ctx => ctx.params.tableId),
|
||||||
|
rowController.patch
|
||||||
|
)
|
||||||
|
.post(
|
||||||
|
"/api/:tableId/rows/validate",
|
||||||
|
authorized(WRITE_TABLE, ctx => ctx.params.tableId),
|
||||||
|
rowController.validate
|
||||||
|
)
|
||||||
|
.delete(
|
||||||
|
"/api/:tableId/rows/:rowId/:revId",
|
||||||
|
authorized(WRITE_TABLE, ctx => ctx.params.tableId),
|
||||||
|
usage,
|
||||||
|
rowController.destroy
|
||||||
|
)
|
||||||
|
|
||||||
|
module.exports = router
|
|
@ -0,0 +1,27 @@
|
||||||
|
const Router = require("@koa/router")
|
||||||
|
const tableController = require("../controllers/table")
|
||||||
|
const authorized = require("../../middleware/authorized")
|
||||||
|
const { BUILDER, READ_TABLE } = require("../../utilities/accessLevels")
|
||||||
|
|
||||||
|
const router = Router()
|
||||||
|
|
||||||
|
router
|
||||||
|
.get("/api/tables", authorized(BUILDER), tableController.fetch)
|
||||||
|
.get(
|
||||||
|
"/api/tables/:id",
|
||||||
|
authorized(READ_TABLE, ctx => ctx.params.id),
|
||||||
|
tableController.find
|
||||||
|
)
|
||||||
|
.post("/api/tables", authorized(BUILDER), tableController.save)
|
||||||
|
.post(
|
||||||
|
"/api/tables/csv/validate",
|
||||||
|
authorized(BUILDER),
|
||||||
|
tableController.validateCSVSchema
|
||||||
|
)
|
||||||
|
.delete(
|
||||||
|
"/api/tables/:tableId/:revId",
|
||||||
|
authorized(BUILDER),
|
||||||
|
tableController.destroy
|
||||||
|
)
|
||||||
|
|
||||||
|
module.exports = router
|
|
@ -2,7 +2,7 @@ const {
|
||||||
createInstance,
|
createInstance,
|
||||||
createClientDatabase,
|
createClientDatabase,
|
||||||
createApplication,
|
createApplication,
|
||||||
createModel,
|
createTable,
|
||||||
createView,
|
createView,
|
||||||
supertest,
|
supertest,
|
||||||
defaultHeaders
|
defaultHeaders
|
||||||
|
@ -12,8 +12,8 @@ const {
|
||||||
generatePowerUserPermissions,
|
generatePowerUserPermissions,
|
||||||
POWERUSER_LEVEL_ID,
|
POWERUSER_LEVEL_ID,
|
||||||
ADMIN_LEVEL_ID,
|
ADMIN_LEVEL_ID,
|
||||||
READ_MODEL,
|
READ_TABLE,
|
||||||
WRITE_MODEL,
|
WRITE_TABLE,
|
||||||
} = require("../../../utilities/accessLevels")
|
} = require("../../../utilities/accessLevels")
|
||||||
|
|
||||||
describe("/accesslevels", () => {
|
describe("/accesslevels", () => {
|
||||||
|
@ -21,7 +21,7 @@ describe("/accesslevels", () => {
|
||||||
let server
|
let server
|
||||||
let request
|
let request
|
||||||
let instanceId
|
let instanceId
|
||||||
let model
|
let table
|
||||||
let view
|
let view
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
|
@ -36,8 +36,8 @@ describe("/accesslevels", () => {
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
instanceId = (await createInstance(request, appId))._id
|
instanceId = (await createInstance(request, appId))._id
|
||||||
model = await createModel(request, appId, instanceId)
|
table = await createTable(request, appId, instanceId)
|
||||||
view = await createView(request, appId, instanceId, model._id)
|
view = await createView(request, appId, instanceId, table._id)
|
||||||
})
|
})
|
||||||
|
|
||||||
describe("create", () => {
|
describe("create", () => {
|
||||||
|
@ -63,7 +63,7 @@ describe("/accesslevels", () => {
|
||||||
it("should list custom levels, plus 2 default levels", async () => {
|
it("should list custom levels, plus 2 default levels", async () => {
|
||||||
const createRes = await request
|
const createRes = await request
|
||||||
.post(`/api/accesslevels`)
|
.post(`/api/accesslevels`)
|
||||||
.send({ name: "user", permissions: [ { itemId: model._id, name: READ_MODEL }] })
|
.send({ name: "user", permissions: [ { itemId: table._id, name: READ_TABLE }] })
|
||||||
.set(defaultHeaders(appId, instanceId))
|
.set(defaultHeaders(appId, instanceId))
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(200)
|
.expect(200)
|
||||||
|
@ -96,7 +96,7 @@ describe("/accesslevels", () => {
|
||||||
it("should delete custom access level", async () => {
|
it("should delete custom access level", async () => {
|
||||||
const createRes = await request
|
const createRes = await request
|
||||||
.post(`/api/accesslevels`)
|
.post(`/api/accesslevels`)
|
||||||
.send({ name: "user", permissions: [ { itemId: model._id, name: READ_MODEL } ] })
|
.send({ name: "user", permissions: [ { itemId: table._id, name: READ_TABLE } ] })
|
||||||
.set(defaultHeaders(appId, instanceId))
|
.set(defaultHeaders(appId, instanceId))
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(200)
|
.expect(200)
|
||||||
|
@ -119,7 +119,7 @@ describe("/accesslevels", () => {
|
||||||
it("should add given permissions", async () => {
|
it("should add given permissions", async () => {
|
||||||
const createRes = await request
|
const createRes = await request
|
||||||
.post(`/api/accesslevels`)
|
.post(`/api/accesslevels`)
|
||||||
.send({ name: "user", permissions: [ { itemId: model._id, name: READ_MODEL }] })
|
.send({ name: "user", permissions: [ { itemId: table._id, name: READ_TABLE }] })
|
||||||
.set(defaultHeaders(appId, instanceId))
|
.set(defaultHeaders(appId, instanceId))
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(200)
|
.expect(200)
|
||||||
|
@ -130,7 +130,7 @@ describe("/accesslevels", () => {
|
||||||
.patch(`/api/accesslevels/${customLevel._id}`)
|
.patch(`/api/accesslevels/${customLevel._id}`)
|
||||||
.send({
|
.send({
|
||||||
_rev: customLevel._rev,
|
_rev: customLevel._rev,
|
||||||
addedPermissions: [ { itemId: model._id, name: WRITE_MODEL } ]
|
addedPermissions: [ { itemId: table._id, name: WRITE_TABLE } ]
|
||||||
})
|
})
|
||||||
.set(defaultHeaders(appId, instanceId))
|
.set(defaultHeaders(appId, instanceId))
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
|
@ -142,8 +142,8 @@ describe("/accesslevels", () => {
|
||||||
.expect(200)
|
.expect(200)
|
||||||
|
|
||||||
expect(finalRes.body.permissions.length).toBe(2)
|
expect(finalRes.body.permissions.length).toBe(2)
|
||||||
expect(finalRes.body.permissions.some(p => p.name === WRITE_MODEL)).toBe(true)
|
expect(finalRes.body.permissions.some(p => p.name === WRITE_TABLE)).toBe(true)
|
||||||
expect(finalRes.body.permissions.some(p => p.name === READ_MODEL)).toBe(true)
|
expect(finalRes.body.permissions.some(p => p.name === READ_TABLE)).toBe(true)
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should remove given permissions", async () => {
|
it("should remove given permissions", async () => {
|
||||||
|
@ -152,8 +152,8 @@ describe("/accesslevels", () => {
|
||||||
.send({
|
.send({
|
||||||
name: "user",
|
name: "user",
|
||||||
permissions: [
|
permissions: [
|
||||||
{ itemId: model._id, name: READ_MODEL },
|
{ itemId: table._id, name: READ_TABLE },
|
||||||
{ itemId: model._id, name: WRITE_MODEL },
|
{ itemId: table._id, name: WRITE_TABLE },
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
.set(defaultHeaders(appId, instanceId))
|
.set(defaultHeaders(appId, instanceId))
|
||||||
|
@ -166,7 +166,7 @@ describe("/accesslevels", () => {
|
||||||
.patch(`/api/accesslevels/${customLevel._id}`)
|
.patch(`/api/accesslevels/${customLevel._id}`)
|
||||||
.send({
|
.send({
|
||||||
_rev: customLevel._rev,
|
_rev: customLevel._rev,
|
||||||
removedPermissions: [ { itemId: model._id, name: WRITE_MODEL }]
|
removedPermissions: [ { itemId: table._id, name: WRITE_TABLE }]
|
||||||
})
|
})
|
||||||
.set(defaultHeaders(appId, instanceId))
|
.set(defaultHeaders(appId, instanceId))
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
|
@ -178,7 +178,7 @@ describe("/accesslevels", () => {
|
||||||
.expect(200)
|
.expect(200)
|
||||||
|
|
||||||
expect(finalRes.body.permissions.length).toBe(1)
|
expect(finalRes.body.permissions.length).toBe(1)
|
||||||
expect(finalRes.body.permissions.some(p => p.name === READ_MODEL)).toBe(true)
|
expect(finalRes.body.permissions.some(p => p.name === READ_TABLE)).toBe(true)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue