Creating an AIMMS Server App
There are several advantages to setting up a client-server architecture:
Independent developments - clients can be developed independently from the server
Scalability - the AIMMS Cloud permits to run several tasks in parallel
AIMMS is designed to create Decision Support Applications, thereby a good choice to specify the actual implementation of a service. Besides actually implementing decision support, to make a service out of your application, you want to take care of the following aspects:
which services should defined,
how to implement the services,
testing the services, and
documenting the services.
This article assumes you are familiar with
Client-server architecture - see also: AIMMS OpenAPI Overview
Using the AIMMS Cloud platform, including using applications and publishing applications - see also: AIMMS Cloud documentation
The application that is used to illustrate various concepts, is just counting the asterisks in a list of strings. As such it is a “Hello World” type of application - just illustrating technology and getting you started to with that technology.
Functionalities of the Application
Let’s start by describing the functionalities of the application:
It counts the number of asterisks in its input.
Generate an input file with some asterisks in it.
The input can be in one of the de facto data standards for providing data:
CSV
Json
Parquet
Excel
XML
For each of the services, the data format of the output is the same as the data format of the input unless there are errors. If there are errors, the data format is always a json file.
For generating sample input files, the output can be in any of the above formats.
As a service is an exact specification of both functionality and input/output data formats, we have ten services to implement. Only one service will be detailed in this article, the others are small variations on that. The details of the other services can be found in the downloads.
Defining the Service
A service is defined by associating a service name with an AIMMS procedure, as illustrated below:
1Procedure pr_countTheStarsJson {
2 Body: {
3 _sp_inp := dex::api::RequestAttribute( 'request-data-path' ) ;
4 _sp_out := dex::api::RequestAttribute( 'response-data-path' ) ;
5
6 pr_actuallyCountStarsJson( _sp_inp, _sp_out );
7
8 return 1;
9 }
10 dex::ServiceName: countStarsJson;
11 StringParameter _sp_inp;
12 StringParameter _sp_out;
13}
Remarks:
Line 10: The annotation
dex::ServiceName
associates the procedurepr_countTheStars
with the servicecountStars
.Lines 3-4: When the procedure is invoked as a task, the string parameter
dex::api::RequestAttribute
is available. Here it is used to pass the names of the input and output files.Line 6: Call the workhorse (see sub section below).
Similar procedures in this example application define the same service, but use other data formats, such as CSV, Excel, Parquet, and XML. In addition, there are similar procedures to generate a sample input file in the various data formats.
Implementing a Service
The actual working code developed is the function fnc_numberOfStars
.
Here the connection is made between that function and the procedure defining the service above.
Procedure pr_actuallyCountStarsJson {
Arguments: (sp_input,sp_output);
Body: {
empty s_lineNumbers ;
dex::ReadFromFile(
dataFile : sp_input,
mappingName : "starsJSON",
emptyIdentifiers : 0,
emptySets : 0,
resetCounters : 0);
p_noStars := fnc_numberOfStars( sp_lines );
! write response body
dex::WriteToFile(
dataFile : sp_output,
mappingName : "countedJSON",
pretty : 0);
! Application specific return code.
return 1;
}
StringParameter sp_input {
Property: Input;
}
StringParameter sp_output {
Property: Input;
}
}
Remarks:
Line 2: The arguments denote the name of the input and output files.
Lines 6-11: Reading of input
Line 13: The actual computation as a simple function call.
Lines 16-19: Writing the output
Tip
The procedure ProfilerStart is called in MainInitialization
enabling tracking task invocations, and task performance.
Testing the Service
There are three types of tests:
In the server app itself, also called unit tests.
On the machine of the AIMMS app developer, using a client app for this purpose.
On the AIMMS Cloud, and using a client app for this purpose.
Performing Unit Tests
An example of a unit test is the following:
Procedure pr_testCountJson {
Body: {
dex::AddMapping(
mappingName : "starsJSON",
mappingFile : "Mappings/starsJSON.xml");
dex::AddMapping(
mappingName : "countedJSON",
mappingFile : "Mappings/countedJSON.xml");
! Call the procedure that does the actual implementation.
pr_actuallyCountStarsJson("data/data.json", "data/noStars.json");
! Verify that the output file has the expected contents.
_sp_jsonContents := FileRead( "data/noStars.json" );
aimmsunit::AssertTrue(
descr : "Expected outcome json",
expr : _sp_jsonContents = "{\"count\":28.0}",
cont : 0);
}
aimmsunit::TestSuite: CountStarsUnitTests;
StringParameter _sp_jsonContents;
}
Such unit tests verify that the server application still has the verified behavior.
More about unit tests can be found at:
Facilitating Tests with Client Apps
To facilitate testing by client apps of the service, the service will need to be:
On local host:
To develop the AIMMS service itself, in AIMMS Developer the service can be started using dex::api::StartAPIService. See also Activating the REST service article.
In the AIMMS Cloud:
After publishing an app
app
with versionver
on the AIMMS Cloud, the service is started when a POST request of the above form is made; there is no need to calldex::api::StartAPIService
from within the service app.
Development Architecture of an AIMMS Service contains a more detailed overview of the communication between the server app and the client apps.
Documenting the Service
For each service, we need to specify its:
Functionality. IIn the running example this would be:
The service
countStarsJson
counts the number of asterisks in a list of strings.Expected input / request body. In the running example this would be:
The expected input is a json file with one member named “lines” and has as value an array of strings.
Output / response body to be expected. In the running example this would be:
The output to be expected is a json file with one member named count, and value the number of asterisks.
Summary
Using the AIMMS language is a good way to define a service atop of a Decision Support application.
With the DataExchange library, defining the interface is essentially a matter of
Selecting input and output formats and linking the contents of these data files to identifiers in the AIMMS application
Selecting the procedure to run
It is good practice to implement unit tests and provide good and detailed documentation of your services.