-
Notifications
You must be signed in to change notification settings - Fork 7
Database API
This API lets you to work and interact with the database layer of M3 Business Engine. Using DatabaseAPI it is possible to perform create, read, update and delete queries and run them. Any changes done on the database will be directly visible to the Business Engine program that the extension is connected to.
Before starting with the code, to understand the database better it is important to get familiar with the different concepts, APIs and tools used to perform an operation on the database.
The main API that is needed to be injected to the extension to perform any database-related operation. All other APIs that are stated below will be retrieved from this API and shall not be declared on top of the extension. The DatabaseAPI is used to define the operations that will be executed on database.
An API which provides you the standard M3's FieldSelection
functionality for database operations. This is used to
define more advanced filters on fields that may or may not be index fields. Since this API provides the ability to
perform filtering on non-indexed fields it should then be used sparingly as it might lead to performance penalties.
A database operation that could be any of create, read, update or delete. The operation contains information about which table, index, table fields and filters will be used to execute the operation.
A container that contains table fields and values. It is used both for setting the keys before executing the operation and also for getting the result after the operation has been executed
An extended version of DBContainer which contains a locked record's fields and values and also methods to perform update and delete operation on this locked record.
To create a record
To read a specific record, primary keys or full index keys should be used to lookup the data. The example below illustrates how to retrieve an item
Example:
Read an item with the index 00 keys
int currentCompany = (Integer)program.getLDAZD().CONO
DBAction query = database.table("MITMAS").index("00").selection("MMCONO", "MMITNO", "MMITDS", "MMSTAT").build()
DBContainer container = query.getContainer()
container.set("MMCONO", currentCompany)
container.set("MMITNO", "SAMPLE-ITEM")
if (query.read(container)) {
String description = container.get("MMITDS")
String status = container.get("MMSTAT")
}
To read multiple records a database action should be defined along with a Closure<?>
that defines how each read record
will be processed/used.
Example:
Read all items with status 20 in a specific company and perform an action on each found record
def handleReleasedItems() {
int currentCompany = (Integer)program.getLDAZD().CONO
DBAction query = database.table("MITMAS").index("20").selection("MMCONO", "MMITNO", "MMITDS", "MMSTAT").build()
DBContainer container = query.getContainer()
container.set("MMCONO", currentCompany)
container.set("MMSTAT", "20")
query.readAll(container, 2, releasedItemProcessor)
}
Closure<?> releasedItemProcessor = { DBContainer container ->
String description = container.get("MMITDS")
String status = container.get("MMSTAT")
// Use this found record as intended
}
Database API has support for expressions and filters, to use the feature, an instance of ExpressionFactory
should be
retrieved to build the filter and then used to build the DBAction. The rest is identical to a normal operation.
Example:
Read all items in current company, status set to 20, item group set to EX-GROUP
and custom field greater than or equal
to 2.
def handleReleasedItems() {
int currentCompany = (Integer)program.getLDAZD().CONO
ExpressionFactory expression = database.getExpressionFactory("MITMAS")
expression.eq("MMITGR", "EX-GROUP").and(expression.gt("MMCFI1", "2"))
DBAction query = database.table("MITMAS").index("20").matching(expression).selection("MMCONO", "MMITNO", "MMITDS", "MMSTAT").build()
DBContainer container = query.getContainer()
container.set("MMCONO", currentCompany)
container.set("MMSTAT", "20")
query.readAll(container, 2, releasedItemProcessor)
}
Closure<?> releasedItemProcessor = { DBContainer container ->
String description = container.get("MMITDS")
String status = container.get("MMSTAT")
// Use this found record as intended
}
To update an table record, the record must be called and update should be performed in callback Closure.
Example:
Read item and update status to 90
def deprecateItem() {
int currentCompany = (Integer)program.getLDAZD().CONO
DBAction query = database.table("MITMAS").index("00").selection("MMCONO", "MMITNO", "MMITDS", "MMSTAT").build()
DBContainer container = query.getContainer()
container.set("MMCONO", currentCompany)
container.set("MMITNO", "SAMPLE-ITEM")
query.readLock(container, updateCallBack)
}
Closure<?> updateCallBack = { LockedResult lockedResult ->
lockedResult.set("MMSTAT", "90")
lockedResult.update()
}
Updating multiple records is not much different from updating once specific record. The only difference would be the method called and the number of keys that are used for reading the records
Example:
Read all items in with status 20 in a company and set the status to 90 for then
def deprecateItems() {
int currentCompany = (Integer)program.getLDAZD().CONO
DBAction query = database.table("MITMAS").index("20").selection("MMCONO", "MMITNO", "MMITDS", "MMSTAT").build()
DBContainer container = query.getContainer()
container.set("MMCONO", currentCompany)
container.set("MMSTAT", "20")
query.readAllLock(container, 2, updateCallBack)
}
Closure<?> updateCallBack = { LockedResult lockedResult ->
lockedResult.set("MMSTAT", "90")
lockedResult.update()
}
Deleting one or several records from the database is no different than update. The only difference would be the update
call which is replaced with delete
instead.
Example:
Delete item with status 90
def deprecateItem() {
int currentCompany = (Integer)program.getLDAZD().CONO
DBAction query = database.table("MITMAS").index("00").selection("MMCONO", "MMITNO", "MMITDS", "MMSTAT").build()
DBContainer container = query.getContainer()
container.set("MMCONO", currentCompany)
container.set("MMITNO", "SAMPLE-ITEM")
query.readLock(container, deleterCallback)
}
Closure<?> deleterCallback = { LockedResult lockedResult ->
lockedResult.delete()
}
The ability to work with database directly is a very tempting solution. It is extremely flexible and yet quite dangerous if it's not handled properly. Keep the following points in mind when working with the DatabaseAPI
- Stick to standard M3 APIs when possible, using direct record insert/update you skip all validations that are done by M3 which might lead to corrupt data in database
- If current program is using the table you are performing operations on, the changes will affect the record read by the program. This might be intentional in some cases. In some other cases you might retrieve another record while the program was processing the original record and it can lead to undefined behaviour
- Avoid using ExpressionFactory and matching filters when possible and leverage defined indexes instead
- Avoid retrieving too many columns, only retrieve the ones you need
- Avoid creating queries that affect large data set. This is not meant to be used for going through thousands of records.
- Avoid performing expensive tasks and calculations while locking a record. Locking a record blocks all other programs in M3 to perform operations on this record
XtendM3 - Customizing M3 Business Logic in cloud
- Introduction
- Getting Started
- API Specification
- Generic APIs
- Integration APIs
- Storage/Persistent APIS
- Context Specific APIs
- Best Practices