Use Mendix Data Storage APIs to Build Reusable Microflow Actions
Introduction
Mendix Studio Pro supports two query languages to retrieve data:
- XPath as an easy to use query language to retrieve objects
- OQL is a SQL based language, more focused on powerful reporting facilities
You can use these query languages in Mendix Studio Pro, but both languages are also available through the Mendix Runtime Java API. You can use this API to implement powerful reusable microflow actions. In addition to XPath and OQL, the Mendix API also enables you to use standard SQL on your Mendix database.
This how to describes how you can build the following microflow actions:
- Retrieve advanced XPath - returns a list of entities as specified by an XPath expression
- Retrieve advanced OQL - returns a list of entities as specified by an OQL query
- Retrieve Dataset OQL - returns a list of entities as specified by a Dataset OQL query
- Retrieve advanced SQL - returns a list of entities as specified by a SQL query
- Create first Monday of month list - returns a list of dates of the first Monday of every month in a specified range
- Register global entity listeners - run custom Java code for every object change
For more information on Java programming for Mendix, see Java Programming.
For more information on calling Java actions from a microflow, see Java Actions.
Retrieving Objects Using Advanced XPath
In this section you will create a microflow action where a user can specify an XPath expression and the result entities which are expected. The action will execute the XPath statement and return the resulting list of objects.
In practice, this is not a very useful microflow action as you can already do this with the standard retrieve action in Mendix Studio Pro. It is used to illustrate how you can use the XPath Java API.
The Java action needs the following parameters:
-
A string where the user can specify the XPath expression to be executed
-
A result entity where the user specifies which entity is to be returned
-
A return type which specifies that the action returns a list of the entities specified in the previous parameter
A type parameter is required to define what object types should be returned in the list. This is specified using the ResultEntity parameter:
Finally, you should define how you want to display the microflow in the microflow toolbox. This consists of a caption, a category and an icon:
The implementation of this Java action is pretty straightforward; you can use the Core.createXPathQuery API to execute your XPath expression and return a list of Mendix objects.
The implementation also validates that the list returned contains objects of the entity specified.
@Override
public java.util.List<IMendixObject> executeAction() throws Exception
{
//BEGIN USER CODE
List<IMendixObject> result = null;
result = Core.createXPathQuery(this.XPath).execute(getContext());
if (!result.isEmpty() && !result.get(0).isInstanceOf(this.ResultEntity)) {
throw new MendixRuntimeException(String.format("Unexpected result entity: %s vs %s",
result.get(0).getMetaObject().getName(), this.ResultEntity));
}
return result;
// END USER CODE
}
Now you have a new microflow action in the toolbox that you can use in your microflows.
Here is an example domain model with two entities: Department and Employee.
You can drag the Java action created above from the toolbox into a microflow. In this example, you want to retrieve all Employee objects and return a list of these objects.
Retrieving Objects Using OQL
The following example illustrates how you can use the OQL APIs for reporting purposes. OQL is the general-purpose Mendix query language, very much resembling SQL. The biggest differences between OQL and SQL are:
- OQL is expressed in entity and attribute names instead of table names and column names. This makes it easier to use, as you do not have to know the technical details of the data model as stored in the database
- OQL is database vendor independent, so you can run the same OQL statement on all databases supported by Mendix
The following non-persistable entity shows what data you are interested in for your report:
- For every department you want to know
- its name,
- the birthday of the oldest employee
- the birthday of the youngest employee
- the total salary bill
- the average salary of the employees
- the minimum salary paid to an employee
Using OQL, you can query this data as follows:
Note that here you use the alias (as …) to map the results of the selection to the attributes in the entity.
You can create a generic microflow action to execute OQL queries and return a list of objects. The Java action has the following parameters:
- OqlQuery – a string containing the OQL query
- ResultEntity – which entity will hold the retrieved data
- A list of the ResultEntity specified as the return type
As in the XPath example above, a Type parameter is defined to specify that the return list uses the type specified in ResultEntity.
Additionally, you need to expose the Java action as a microflow action, so provide a caption and an icon.
The Java action illustrated below does the following:
-
Retrieves all data using the Mendix API
Core.retrieveOQLDataTable()
-
Loops through all the rows, creates a new object of the type specified by ResultEntity
Setting a Java action parameter of type Entity of type parameter… (ResultEntity in the example above) creates a Java string in the action which contains the name of the entity type. This string can be passed to Core.instantiate to create a new object. -
Loops through all columns of a record and copies the column value to an attribute with the same name. If an attribute with the specified column name does not exist, a message is printed and the loop continues
-
The Mendix object created is added to the list to be returned
Note that in this case, as show in the domain model screenshot and the OQL screenshot above, the names of the attributes and columns match exactly. (See RetrieveAdvancedOql.java in the QueryApiBlogPost GitHub repo for the full source code).
@Override
public java.util.List<IMendixObject> executeAction() throws Exception
{
// BEGIN USER CODE
List<IMendixObject> resultList = new ArrayList<IMendixObject>();
IDataTable resultDT = Core.retrieveOQLDataTable(getContext(), this.OqlQuery);
int colCount = resultDT.getSchema().getColumnCount();
// Loop through records, add to mendix object list
resultDT.forEach(row -> {
// instantiate mendix object as specified by ResultEnitity parameter
IMendixObject obj = Core.instantiate(getContext(), this.ResultEntity);
for (int i = 0; i < colCount; i++) {
// get column name
String colName = resultDT.getSchema().getColumnSchema(i).getName();
// get column value
Object colValue = row.getValue(getContext(), i);
if(obj.hasMember(colName)) {
// set result object value
obj.setValue(getContext(), colName, colValue);
} else {
logger.info(String.format("Target entity does not have attribute named %s",colName));
}
}
resultList.add(obj);
});
return resultList;
// END USER CODE
}
The result is a generic OQL action that you can use in your microflows as follows:
Retrieving Objects Using OQL Specified in a Dataset
Instead of coding the OQL statement in a string parameter, you can also use a Dataset. This has the benefit that Mendix Studio Pro will validate your OQL query.
This time, you need to define a Java action that will take the name of the dataset. This action will get the OQL from the DataSet, execute it, and return a list of Mendix objects.
The microflow to execute the Java action is similar to the previous example, but instead of an OQL query, you specify the name of the Dataset.
Below is the Java code to get the Dataset OQL, execute the OQL, and retrieve the Objects. You use the Core.createOQLTextGetRequestFromDataSet method to get the OQL query of the Dataset specified. (See RetrieveDatasetOql.java in the QueryApiBlogPost GitHub repo for the full source code).
@Override
public java.util.List<IMendixObject> executeAction() throws Exception
{
// BEGIN USER CODE
ILogNode logger = Core.getLogger("RetrieveDatasetOql");
List<IMendixObject> resultList = new ArrayList<IMendixObject>();
IOQLTextGetRequest oqlGetRequest = Core.createOQLTextGetRequestFromDataSet(this.DataSetName);
String oqlQuery = oqlGetRequest.getQuery();
logger.info("OQL: " + oqlQuery);
IDataTable resultDT = Core.retrieveOQLDataTable(getContext(), oqlQuery);
int colCount = resultDT.getSchema().getColumnCount();
resultDT.forEach(row -> {
logger.info("Row: " + row.getValue(getContext(), 0));
IMendixObject obj = Core.instantiate(getContext(), this.ResultEntity);
for (int i = 0; i < colCount; i++) {
String colName = resultDT.getSchema().getColumnSchema(i).getName();
Object colValue = row.getValue(getContext(), i);
if (obj.hasMember(colName)) {
obj.setValue(getContext(), colName, colValue);
} else {
logger.info(String.format("Target entity %s does not have attribute named %s", this.ResultEntity, colName));
}
}
resultList.add(obj);
});
return resultList;
// END USER CODE
}
Retrieving Objects Using SQL
You can use an API call to execute SQL queries on the application database. Using this call, you can create a microflow action to execute SQL: similar to the action for OQL in the previous sections.
The definition of the Java action resembles the OQL action, but instead of an OQL parameter you have an SQL parameter.
The Java code below implements the following steps:
-
Use Core.dataStorage().executeWithConnection() to execute some Java statements that receive a JDBC connection from the internal connection pool. Using this call enables the Mendix Platform to guarantee that connections are returned to the pool after use.
@Override public java.util.List<IMendixObject> executeAction() throws Exception { // BEGIN USER CODE logger.info("executeAction: " + this.Sql); List<IMendixObject> resultList = null; resultList = Core.dataStorage().executeWithConnection(connection -> {...}); return resultList; // END USER CODE }
-
With the JDBC connection you can now write your Java as you would with a regular JDBC connection.
-
A prepared statement is created, executed, and the resulting records are made available through a
ResultSet
.resultList = Core.dataStorage().executeWithConnection(connection -> { List<IMendixObject> result = new ArrayList<IMendixObject>(); try { PreparedStatement stmt = connection.prepareStatement(this.Sql); ResultSet rset = stmt.executeQuery(); ResultSetMetaData rmd = rset.getMetaData(); int colCount = rmd.getColumnCount(); ```
-
Next you loop through all the records in the
ResultSet
and, for each record, create a Mendix object as specified by the user via ResultEntity.while(rset.next()){ IMendixObject obj = Core.instantiate(getContext(),this.ResultEntity); result.add(obj); for(int colIdx=1; colIdx <= colCount ; colIdx++){ String colName = rmd.getColumnName(colIdx); obj.setValue(getContext(),colName,rset.getObject(colIdx)); } logger.debug(String.format("Created object %s", obj)); } ```
You can find the complete Java source code in the QueryApiBlogPost GitHub repo on GitHub: RetrieveAdvancedSQL.
You now have a generic SQL action that can be used in microflows to retrieve data from your application database. The query in this example returns the same data as the OQL earlier, so you can reuse the non-persistable entity DepartmentSummary as defined previously.
PostgreSQL-specific SQL
Using the JDBC connection you can benefit from vendor specific database extensions, like Oracle Pl/SQL or Postgres user-defined functions.
The following example illustrates the use of PostgreSQL-specific functionality. It serves as an example of how you can do this, but in this specific case an alternative solution, either using microflows or Java actions, would be better as it would keep your application database independent.
The requirement for this example is to generate a list of dates for all the first Mondays of the month within a range specified by the user.
This example has a page where an end-user can enter a start and end date. The microflow triggered by the Generate first Mondays of the month button will write all the respective dates to the log.
In Postgres you can query a list of the dates of all Mondays between these dates using the following Postgres specific query:
- Using a common table expression (CTE), you create a set of all first dates of every month in the range
- Using another CTE you determine the dates of the Mondays for these months
- Finally, you select these dates if they still fall in range specified
For example:
with first_day_of_month as (
SELECT *
FROM generate_series
( date_trunc('month', '2017-01-24 00:00'::timestamp)
, '2017-11-05 12:00', '1 months'
) as firstday
),
firstmonday as (
select fdom.firstday::date +
((8 - extract(dow from fdom.firstday))::integer % 7)
as first_monday_date
from first_day_of_month as fdom
)
select fm.first_monday_date
from firstmonday as fm
where fm.first_monday_date >= '2017-01-24 00:00'::timestamp
and fm.first_monday_date <= '2017-11-05 12:00'::timestamp
;
Creating the Java Action
You create a Java action with parameters for the start date and the end date. You have a specific entity to return a list of the dates: Hr.FirstMondayDate.
Creating the Java Code
-
Specify the required SQL statement in the Java method. JDBC queries expect the parameters to be specified by question marks (?) in the SQL statement.
@Override public java.util.List<IMendixObject> executeAction() throws Exception { // BEGIN USER CODE String sql = "with first_day_of_month as (\n " + " SELECT * \n" + " FROM generate_series\n" + " ( date_trunc('month', ?::timestamp)\n" + " , ?, '1 months'\n" + " ) as firstday\n" + "),\n" + "firstmonday as (\n" + " select fdom.firstday::date + \n" + " ((8 - extract(dow from fdom.firstday))::integer % 7) \n" + " as first_monday_date\n" + " from first_day_of_month as fdom\n" + ")\n" + "select fm.first_monday_date\n" + "from firstmonday as fm\n" + "where fm.first_monday_date >= ?::timestamp\n" + "and fm.first_monday_date <= ?::timestamp\n" + ";" ; logger.info("executeAction: " + sql);
-
Next, use the Mendix API to execute some statements using the JDBC connection. Here you create a prepared statement, define the JDBC parameter values, and execute the SQL query.
List<IMendixObject> resultList = null; resultList = Core.dataStorage().executeWithConnection(connection -> { List<IMendixObject> result = new ArrayList<IMendixObject>(); try { PreparedStatement stmt = connection.prepareStatement(sql); // bind start and end date variables stmt.setDate(1,new java.sql.Date(this.StartDate.getTime())); stmt.setDate(2, new java.sql.Date(this.EndDate.getTime())); stmt.setDate(3,new java.sql.Date(this.StartDate.getTime())); stmt.setDate(4, new java.sql.Date(this.EndDate.getTime())); ResultSet rset = stmt.executeQuery(); ResultSetMetaData rmd = rset.getMetaData();
-
Using the
FirstMondayDate
Java proxy, instantiate a new Mendix object and set the date attribute. -
Finally, return the created list of dates.
// loop through first monday records while(rset.next()){ // create FirstMondayDate mendix entity and add to list FirstMondayDate dateObj = new hr.proxies.FirstMondayDate(getContext()); result.add(dateObj.getMendixObject()); dateObj.setDate(rset.getDate(1)); logger.debug(String.format("Created object %s", dateObj)); } } catch (SQLException e) { logger.error("Failed to execute sql statement: " + e.getMessage()); throw new MendixRuntimeException(e); } return result; }); return resultList; // END USER CODE
When you use this in a microflow, you just need to specify the start and end dates, and the name of the resulting list. This example iterates through all the data objects in the list and writes the date of that object to the log.
You will see the list of dates in the console.
Global Custom Entity Event Listeners
Global entity event listeners enable you to define generic event handlers on all entities. This enables you to build generic validations or create a real-time data export to a central datastore. You can use a Java action to register any desired event handler, most likely in the After App Startup Microflow.
Example code to Register the Event Listener
This code will log old and new attribute value for all changes to attributes before making changes in the database:
public java.lang.Boolean executeAction() throws Exception {
// BEGIN USER CODE
Core.getListenersRegistry().registerBeforeCommitListener(objects -> {
ILogNode logger = Core.getLogger("BeforeCommitListener");
for (IMendixObject obj : objects) {
logger.info("ObjectType: " + obj.getType());
List<? extends IMendixObjectMember<?>> changedMembers = obj.getChangedMembers(getContext());
logger.info(String.format("Has changed members: %b? Number of changed members: %d", obj.isChanged(), changedMembers.size()));
for (IMendixObjectMember member : changedMembers) {
logger.info(
String.format("Changed member %s : %s -> %s", member.getName(),
member.getOriginalValue(getContext()) != null ? member.getOriginalValue(getContext()).toString() : "",
member.getValue(getContext()) != null ? member.getValue(getContext()).toString() : ""
)
);
}
}
});
return true;
// END USER CODE
}
This example will trigger a listener for every object change before writing the changes to the database. To find out what attributes have been changed, you can use the getChangedMembers
method, as illustrated above.