fix delete functionality
This commit is contained in:
parent
480a73997d
commit
da32029c31
|
@ -63,6 +63,212 @@ exports.patch = async function (ctx) {
|
||||||
}
|
}
|
||||||
|
|
||||||
exports.save = async function (ctx) {
|
exports.save = async function (ctx) {
|
||||||
|
if (ctx.request.body.type === 'delete') {
|
||||||
|
await bulkDelete(ctx)
|
||||||
|
} else {
|
||||||
|
await saveRecords(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
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async function bulkDelete(ctx) {
|
||||||
|
const { records } = ctx.request.body
|
||||||
|
const db = new CouchDB(ctx.user.instanceId)
|
||||||
|
|
||||||
|
await db.bulkDocs(
|
||||||
|
records.map(record => ({ ...record, _deleted: true }), (err, res) => {
|
||||||
|
if (err) {
|
||||||
|
ctx.status = 500
|
||||||
|
} else {
|
||||||
|
records.forEach(record => {
|
||||||
|
emitEvent(`record:delete`, ctx, record)
|
||||||
|
})
|
||||||
|
ctx.status = 200
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
async function saveRecords(ctx) {
|
||||||
const instanceId = ctx.user.instanceId
|
const instanceId = ctx.user.instanceId
|
||||||
const db = new CouchDB(instanceId)
|
const db = new CouchDB(instanceId)
|
||||||
let record = ctx.request.body
|
let record = ctx.request.body
|
||||||
|
@ -71,300 +277,94 @@ exports.save = async function (ctx) {
|
||||||
if (!record._rev && !record._id) {
|
if (!record._rev && !record._id) {
|
||||||
record._id = generateRecordID(record.modelId)
|
record._id = generateRecordID(record.modelId)
|
||||||
}
|
}
|
||||||
exports.save = async function (ctx) {
|
|
||||||
const db = new CouchDB(ctx.user.instanceId)
|
|
||||||
let record = ctx.request.body
|
|
||||||
record.modelId = ctx.params.modelId
|
|
||||||
|
|
||||||
if (!record._rev && !record._id) {
|
const model = await db.get(record.modelId)
|
||||||
record._id = generateRecordID(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
|
||||||
|
}
|
||||||
|
|
||||||
const model = await db.get(record.modelId)
|
const existingRecord = record._rev && (await db.get(record._id))
|
||||||
|
|
||||||
record = coerceRecordValues(record, model)
|
// make sure link records are up to date
|
||||||
|
record = await linkRecords.updateLinks({
|
||||||
|
instanceId,
|
||||||
|
eventType: linkRecords.EventType.RECORD_SAVE,
|
||||||
|
record,
|
||||||
|
modelId: record.modelId,
|
||||||
|
model,
|
||||||
|
})
|
||||||
|
|
||||||
const validateResult = await validate({
|
if (existingRecord) {
|
||||||
record,
|
const response = await db.put(record)
|
||||||
model,
|
|
||||||
})
|
|
||||||
|
|
||||||
if (!validateResult.valid) {
|
|
||||||
ctx.status = 400
|
|
||||||
ctx.body = {
|
|
||||||
status: 400,
|
|
||||||
errors: validateResult.errors,
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const existingRecord = record._rev && (await db.get(record._id))
|
|
||||||
|
|
||||||
// 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
|
record._rev = response.rev
|
||||||
|
record.type = "record"
|
||||||
ctx.eventEmitter &&
|
|
||||||
ctx.eventEmitter.emitRecord(`record:save`, instanceId, record, model)
|
|
||||||
ctx.body = record
|
ctx.body = record
|
||||||
ctx.status = 200
|
ctx.status = 200
|
||||||
ctx.message = `${model.name} created successfully`
|
ctx.message = `${model.name} updated successfully.`
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
exports.fetchView = async function (ctx) {
|
record.type = "record"
|
||||||
const instanceId = ctx.user.instanceId
|
const response = await db.post(record)
|
||||||
const db = new CouchDB(instanceId)
|
record._rev = response.rev
|
||||||
const { stats, group, field } = ctx.query
|
|
||||||
const viewName = ctx.params.viewName
|
|
||||||
|
|
||||||
// if this is a model view being looked for just transfer to that
|
ctx.eventEmitter &&
|
||||||
if (viewName.indexOf(MODEL_VIEW_BEGINS_WITH) === 0) {
|
ctx.eventEmitter.emitRecord(`record:save`, instanceId, record, model)
|
||||||
ctx.params.modelId = viewName.substring(4)
|
ctx.body = record
|
||||||
await exports.fetchModelRecords(ctx)
|
ctx.status = 200
|
||||||
return
|
ctx.message = `${model.name} created successfully`
|
||||||
}
|
}
|
||||||
|
|
||||||
const response = await db.query(`database/${viewName}`, {
|
const TYPE_TRANSFORM_MAP = {
|
||||||
include_docs: !stats,
|
link: {
|
||||||
group,
|
"": [],
|
||||||
})
|
[null]: [],
|
||||||
|
[undefined]: undefined,
|
||||||
if (stats) {
|
},
|
||||||
response.rows = response.rows.map(row => ({
|
options: {
|
||||||
group: row.key,
|
"": "",
|
||||||
field,
|
[null]: "",
|
||||||
...row.value,
|
[undefined]: undefined,
|
||||||
avg: row.value.sum / row.value.count,
|
},
|
||||||
}))
|
string: {
|
||||||
} else {
|
"": "",
|
||||||
response.rows = response.rows.map(row => row.doc)
|
[null]: "",
|
||||||
}
|
[undefined]: undefined,
|
||||||
|
},
|
||||||
ctx.body = await linkRecords.attachLinkInfo(instanceId, response.rows)
|
number: {
|
||||||
}
|
"": null,
|
||||||
|
[null]: null,
|
||||||
exports.fetchModelRecords = async function (ctx) {
|
[undefined]: undefined,
|
||||||
const instanceId = ctx.user.instanceId
|
parse: n => parseFloat(n),
|
||||||
const db = new CouchDB(instanceId)
|
},
|
||||||
const response = await db.allDocs(
|
datetime: {
|
||||||
getRecordParams(ctx.params.modelId, null, {
|
"": null,
|
||||||
include_docs: true,
|
[undefined]: undefined,
|
||||||
})
|
[null]: null,
|
||||||
)
|
},
|
||||||
ctx.body = response.rows.map(row => row.doc)
|
attachment: {
|
||||||
ctx.body = await linkRecords.attachLinkInfo(
|
"": [],
|
||||||
instanceId,
|
[null]: [],
|
||||||
response.rows.map(row => row.doc)
|
[undefined]: undefined,
|
||||||
)
|
},
|
||||||
}
|
boolean: {
|
||||||
|
"": null,
|
||||||
exports.search = async function (ctx) {
|
[null]: null,
|
||||||
const instanceId = ctx.user.instanceId
|
[undefined]: undefined,
|
||||||
const db = new CouchDB(instanceId)
|
true: true,
|
||||||
const response = await db.allDocs({
|
false: false,
|
||||||
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)
|
|
||||||
}
|
|
||||||
|
|
||||||
async function bulkDelete(ctx) {
|
|
||||||
const { records } = ctx.request.body
|
|
||||||
const db = new CouchDB(ctx.user.instanceId)
|
|
||||||
|
|
||||||
await db.bulkDocs(
|
|
||||||
records.map(record => ({ ...record, _deleted: true }), (err, res) => {
|
|
||||||
if (err) {
|
|
||||||
ctx.status = 500
|
|
||||||
} else {
|
|
||||||
records.forEach(record => {
|
|
||||||
emitEvent(`record:delete`, ctx, record)
|
|
||||||
})
|
|
||||||
ctx.status = 200
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
|
|
||||||
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,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
Loading…
Reference in New Issue