LINQ Reporting Engine API
Building Reports
To build a report from a template, you can use one of the ReportingEngine.build_report overloads. The following table describes the parameters of these overloads.
Parameter | Description |
---|---|
document |
A template document. At runtime, this document instance is populated with a data from the specified source and becomes a ready report. |
data_source |
An object providing a data to populate the specified template. The object must be of one of the following types: - An XmlDataSource instance (see “Accessing XML Data” for more information) - A JsonDataSource instance (see “Accessing JSON Data” for more information) - A CsvDataSource instance (see “Accessing CSV Data” for more information) |
data_source_name |
The identifier of the specified data source object within the specified template. You can omit this identifier, if the template uses the contextual object member access (see “ REF contextual Using Contextual Object Member Access” for more information) when dealing with the data source. |
Accessing XML Data
You can pass an XmlDataSource instance to the engine as a data source instead. Even when XML schema is not provided, XmlDataSource is capable to recognize values of the following types by their string representations:
int
float
bool
datetime
Note – For recognition of data types to work, string representations of corresponding attributes and text values of XML elements must be formed using invariant culture settings.
While loading data to XmlDataSource, the engine performs actions typical for XML deserialization behind the scenes: It maps complex-type XML elements to internal objects and simple-type XML elements to fields of containing objects. So, in template documents, an XmlDataSource instance should be treated as an object having corresponding fields and nested objects as shown in the following example.
XML
<Person>
<Name>John Doe</Name>
<Age>30</Age>
<Birth>1989-04-01 4:00:00 pm</Birth>
<Child>Ann Doe</Child>
<Child>Charles Doe</Child>
</Person>
Template document
Name: <<[Name]>>, Age: <<[Age]>>, Date of Birth:
<<[Birth]:"dd.MM.yyyy">>
Children:
<<foreach [in Child]>><<[Child_Text]>>
<</foreach>>
Source code
doc = ... # Loading a template document.
dataSource = ... # Loading XML.
engine = aw.reporting.ReportingEngine()
engine.build_report(doc, dataSource)
Result document
Name: John Doe, Age: 30, Date of Birth: 01.04.1989
Children:
Ann Doe
Charles Doe
Note – To reference a sequence of repeated simple-type XML elements with the same name, the elements’ name itself (for example, “Child”) should be used in a template document, whereas the same name with the “_Text” suffix (for example, “Child_Text”) should be used to reference the text value of one of these elements.
By default, if a root XML element contains only a sequence of elements of one type, the engine does not generate an internal root object while loading XML data. So, in template documents, such an XmlDataSource instance should be treated as a sequence of corresponding nested objects as shown in the following example.
XML
<Persons>
<Person>
<Name>John Doe</Name>
<Age>30</Age>
<Birth>1989-04-01 4:00:00 pm</Birth>
</Person>
<Person>
<Name>Jane Doe</Name>
<Age>27</Age>
<Birth>1992-01-31 07:00:00 am</Birth>
</Person>
<Person>
<Name>John Smith</Name>
<Age>51</Age>
<Birth>1968-03-08 1:00:00 pm</Birth>
</Person>
</Persons>
Template document
<<foreach [in persons]>>Name: <<[Name]>>, Age: <<[Age]>>, Date of Birth: <<[Birth]:"dd.MM.yyyy">>
<</foreach>>
Average age: <<[persons.Average(p => p.Age)]>>
Source code
doc = ... # Loading a template document.
dataSource = ... # Loading XML.
engine = aw.reporting.ReportingEngine()
engine.build_report(doc, dataSource, "persons")
Result document
Name: John Doe, Age: 30, Date of Birth: 01.04.1989
Name: Jane Doe, Age: 27, Date of Birth: 31.01.1992
Name: John Smith, Age: 51, Date of Birth: 08.03.1968
Average age: 36
However, if your scenario requires an internal object for a root XML element to be always generated while loading data to XmlDataSource, you can force this as shown in the following code snippet.
options = aw.reporting.XmlDataLoadOptions()
options.always_generate_root_object = True
dataSource = aw.reporting.XmlDataSource(..., options)
The following example sums up typical scenarios involving nested complex-type XML elements.
XML
<Managers>
<Manager>
<Name>John Smith</Name>
<Contract>
<Client>
<Name>A Company</Name>
</Client>
<Price>1200000</Price>
</Contract>
<Contract>
<Client>
<Name>B Ltd.</Name>
</Client>
<Price>750000</Price>
</Contract>
<Contract>
<Client>
<Name>C & D</Name>
</Client>
<Price>350000</Price>
</Contract>
</Manager>
<Manager>
<Name>Tony Anderson</Name>
<Contract>
<Client>
<Name>E Corp.</Name>
</Client>
<Price>650000</Price>
</Contract>
<Contract>
<Client>
<Name>F & Partners</Name>
</Client>
<Price>550000</Price>
</Contract>
</Manager>
<Manager>
<Name>July James</Name>
<Contract>
<Client>
<Name>G & Co.</Name>
</Client>
<Price>350000</Price>
</Contract>
<Contract>
<Client>
<Name>H Group</Name>
</Client>
<Price>250000</Price>
</Contract>
<Contract>
<Client>
<Name>I & Sons</Name>
</Client>
<Price>100000</Price>
</Contract>
<Contract>
<Client>
<Name>J Ent.</Name>
</Client>
<Price>100000</Price>
</Contract>
</Manager>
</Managers>
Template document
<<foreach [in managers]>>Manager: <<[Name]>>
Contracts:
<<foreach [in Contract]>>- <<[Client.Name]>> ($<<[Price]>>)
<</foreach>>
<</foreach>>
Source code
doc = ... # Loading a template document.
dataSource = ... # Loading XML.
engine = aw.reporting.ReportingEngine()
engine.build_report(doc, dataSource, "managers")
Result document
Manager: John Smith
Contracts:
- A Company ($1200000)
- B Ltd. ($750000)
- C & D ($350000)
Manager: Tony Anderson
Contracts:
- E Corp. ($650000)
- F & Partners ($550000)
Manager: July James
Contracts:
- G & Co. ($350000)
- H Group ($250000)
- I & Sons ($100000)
- J Ent. ($100000)
Accessing JSON Data
To access JSON data while building a report, you can pass a JsonDataSource instance to the engine as a data source.
Using of JsonDataSource enables you to work with typed values of JSON elements in template documents. For more convenience, the set of simple JSON types is extended as follows:
int
float
bool
datetime
str
Note – Working with complex JSON types (objects and arrays) is also supported.
While loading data to JsonDataSource, the engine performs JSON deserialization and generates corresponding internal objects. So, in template documents, a JsonDataSource instance should be treated according to what a root JSON element represents.
If a root JSON element is an object, a JsonDataSource instance should be treated as an object as well as shown in the following example.
JSON
{
Name: "John Doe",
Age: 30,
Birth: "1989-04-01 4:00:00 pm",
Child: [ "Ann Doe", "Charles Doe" ]
}
Template document
Name: <<[Name]>>, Age: <<[Age]>>, Date of Birth: <<[Birth]:"dd.MM.yyyy">>
Children:
<<foreach [in Child]>><<[Child_Text]>>
<</foreach>>
Source code
doc = ... # Loading a template document.
dataSource = ... # Loading JSON.
engine = aw.reporting.ReportingEngine()
engine.build_report(doc, dataSource)
Result document
Name: John Doe, Age: 30, Date of Birth: 01.04.1989
Children:
Ann Doe
Charles Doe
Note – To reference a JSON
object property that is an array of simple-type values, the name of the property (for example, “Child”) should be used in a template document, whereas the same name with the “_Text” suffix (for example, “Child_Text”) should be used to reference the value of an item of this array.
If a root JSON element is an array, a JsonDataSource instance should be treated as a sequence of items of this array as shown in the following example.
JSON
[
{
Name: "John Doe",
Age: 30,
Birth: "1989-04-01 4:00:00 pm"
},
{
Name: "Jane Doe",
Age: 27,
Birth: "1992-01-31 07:00:00 am"
},
{
Name: "John Smith",
Age: 51,
Birth: "1968-03-08 1:00:00 pm"
}
]
Template document
<<foreach [in persons]>>Name: <<[Name]>>, Age: <<[Age]>>, Date of Birth: <<[Birth]:"dd.MM.yyyy">>
<</foreach>>
Average age: <<[persons.Average(p => p.Age)]>>
Source code
doc = ... # Loading a template document.
dataSource = ... # Loading JSON.
engine = aw.reporting.ReportingEngine()
engine.build_report(doc, dataSource, "persons")
Result document
Name: John Doe, Age: 30, Date of Birth: 01.04.1989
Name: Jane Doe, Age: 27, Date of Birth: 31.01.1992
Name: John Smith, Age: 51, Date of Birth: 08.03.1968
Average age: 36
By default, if a root JSON element is an object having only one property that is an object or array in turn, the engine does not generate an internal root object while loading JSON data. So, in template documents, such a JsonDataSource instance should be treated according to what this property represents instead. For instance, the following JSON snippets can be used to produce the same results in previous examples of this section respectively.
JSON 1
{
Person:
{
Name: "John Doe",
Age: 30,
Birth: "1989-04-01 4:00:00 pm",
Child: [ "Ann Doe", "Charles Doe" ]
}
}
JSON 2
{
Persons:
[
{
Name: "John Doe",
Age: 30,
Birth: "1989-04-01 4:00:00 pm"
},
{
Name: "Jane Doe",
Age: 27,
Birth: "1992-01-31 07:00:00 am"
},
{
Name: "John Smith",
Age: 51,
Birth: "1968-03-08 1:00:00 pm"
}
]
}
However, if your scenario requires an internal object for a root JSON element to be always generated while loading data to JsonDataSource, you can force this as shown in the following code snippet.
options = aw.reporting.JsonDataLoadOptions()
options.always_generate_root_object = True
dataSource = aw.reporting.JsonDataSource(..., options)
The following example sums up typical scenarios involving nested JSON objects and arrays.
JSON
[
{
Name: "John Smith",
Contract:
[
{
Client:
{
Name: "A Company"
},
Price: 1200000
},
{
Client:
{
Name: "B Ltd."
},
Price: 750000
},
{
Client:
{
Name: "C & D"
},
Price: 350000
}
]
},
{
Name: "Tony Anderson",
Contract:
[
{
Client:
{
Name: "E Corp."
},
Price: 650000
},
{
Client:
{
Name: "F & Partners"
},
Price: 550000
}
]
},
{
Name: "July James",
Contract:
[
{
Client:
{
Name: "G & Co."
},
Price: 350000
},
{
Client:
{
Name: "H Group"
},
Price: 250000
},
{
Client:
{
Name: "I & Sons"
},
Price: 100000
},
{
Client:
{
Name: "J Ent."
},
Price: 100000
}
]
}
]
Template document
<<foreach [in managers]>>Manager: <<[Name]>>
Contracts:
<<foreach [in Contract]>>- <<[Client.Name]>> ($<<[Price]>>)
<</foreach>>
<</foreach>>
Source code
doc = ... # Loading a template document.
dataSource = ... # Loading JSON.
engine = aw.reporting.ReportingEngine()
engine.build_report(doc, dataSource, "managers")
Result document
Manager: John Smith
Contracts:
- A Company ($1200000)
- B Ltd. ($750000)
- C & D ($350000)
Manager: Tony Anderson
Contracts:
- E Corp. ($650000)
- F & Partners ($550000)
Manager: July James
Contracts:
- G & Co. ($350000)
- H Group ($250000)
- I & Sons ($100000)
- J Ent. ($100000)
For recognition of JSON simple values (null, boolean, number, integer, and string), the engine provides two modes: loose and strict. In the loose mode, types of JSON simple values are determined upon parsing of their string representations. In the strict mode, types of JSON simple values are determined from JSON notation itself. To see the main difference between the modes, consider the following JSON snippet.
{ prop: "123" }
In the loose mode, the type of prop
is determined as an integer, whereas in the strict mode, it is determined as a string.
The loose mode is used by the engine by default to support more typed data representation options. However, in some scenarios, it can be more preferable to disable recognition of numbers and other JSON simple values from strings, for example, when you need to keep leading padding zeros in a string value representing a number. In this case, you can switch to the strict mode as shown in the following code snippet.
options = aw.reporting.JsonDataLoadOptions()
options.simple_value_parse_mode = aw.reporting.JsonSimpleValueParseMode.STRICT
dataSource = aw.reporting.JsonDataSource(..., options)
Note – Parsing of date-time values does not depend on whether the loose or strict mode is used.
Recognition of date-time values is a special case, because JSON specification does not define a format for their representation. So, by default, while parsing date-time values from strings, the engine tries several formats in the following order:
- The ISO-8601 format (for values like “2015-03-02T13:56:04Z”)
- The Microsoft® JSON date-time format (for values like “/Date(1224043200000)/")
- All date-time formats supported for the current culture
- All date-time formats supported for the English USA culture
- All date-time formats supported for the English New Zealand culture
Although this approach is quite flexible, in some scenarios, you may need to restrict strings to be recognized as date-time values. You can achieve this by specifying one or several exact formats in the context of the current culture to be used while parsing date-time values from strings as shown in the following example.
formats = [ "MM/dd/yyyy" ]
options = aw.reporting.JsonDataLoadOptions()
options.exact_date_time_parse_formats = formats
dataSource = aw.reporting.JsonDataSource(..., options)
In this example, strings conforming the format “MM/dd/yyyy” are going to be recognized as date-time values while loading JSON, whereas the others are not (but see the following note).
In some scenarios, you may need to disable recognition of date-time values at all, for example, when you deal with strings containing already formatted date-time values, which you do not want to re-format using the engine. You can achieve this by setting the exact date-time parse formats to an empty list (but see the following note).
Note – Strings conforming the Microsoft® JSON date-time format (for example, “/Date(1224043200000)/") are always recognized as date-time values regardless of the exact date-time parse format.
Accessing CSV Data
To access CSV data while building a report, you can pass a CsvDataSource instance to the engine as a data source.
Using of CsvDataSource enables you to work with typed values rather than just strings in template documents. Although CSV as a format does not define a way to store values of types other than strings, CsvDataSource is capable to recognize values of the following types by their string representations:
int
float
bool
datetime
Note – For recognition of data types to work, string representations of corresponding values must be formed using invariant culture settings.
In template documents, a CsvDataSource instance should be treated as a sequence of objects having corresponding fields as shown in the following example.
CSV
John Doe,30,1989-04-01 4:00:00 pm
Jane Doe,27,1992-01-31 07:00:00 am
John Smith,51,1968-03-08 1:00:00 pm
Template document
<<foreach [in persons]>>Name: <<[Column1]>>, Age: <<[Column2]>>, Date of Birth: <<[Column3]:"dd.MM.yyyy">>
<</foreach>>
Average age: <<[persons.Average(p => p.Column2)]>>
Source code
doc = ... # Loading a template document.
dataSource = ... # Loading CSV.
engine = aw.reporting.ReportingEngine()
engine.build_report(doc, dataSource, "persons")
Result document
Name: John Doe, Age: 30, Date of Birth: 01.04.1989
Name: Jane Doe, Age: 27, Date of Birth: 31.01.1992
Name: John Smith, Age: 51, Date of Birth: 08.03.1968
Average age: 36
By default, CsvDataSource uses column names such as “Column1”, “Column2”, and so on, as you can see from the previous example. However, CsvDataSource can be configured to read column names from the first line of CSV data as shown in the following example.
CSV
Name,Age,Birth
John Doe,30,1989-04-01 4:00:00 pm
Jane Doe,27,1992-01-31 07:00:00 am
John Smith,51,1968-03-08 1:00:00 pm
Template document
<<foreach [in persons]>>Name: <<[Name]>>, Age: <<[Age]>>, Date of Birth: <<[Birth]:"dd.MM.yyyy">>
<</foreach>>
Average age: <<[persons.Average(p => p.Age)]>>
Source code
doc = ... # Loading a template document.
options = aw.reporting.CsvDataLoadOptions(True)
dataSource = aw.reporting.CsvDataSource(..., options) # Loading CSV.
engine = aw.reporting.ReportingEngine()
engine.build_report(doc, dataSource, "persons")
Result document
Name: John Doe, Age: 30, Date of Birth: 01.04.1989
Name: Jane Doe, Age: 27, Date of Birth: 31.01.1992
Name: John Smith, Age: 51, Date of Birth: 08.03.1968
Average age: 36
Also, you can use CsvDataLoadOptions to customize the following characters playing special roles while loading CSV data:
- Value separator (the default is comma)
- Single-line comment start (the default is sharp)
- Quotation mark enabling to use other special characters within a value (the default is double quotes)
Removing Paragraphs Containing Only Template Syntax Tags
While building a report, some paragraphs containing only template syntax tags can become empty after the tags are removed or replaced with empty values. To remove such paragraphs from the report, you can apply the ReportBuildOptions.REMOVE_EMPTY_PARAGRAPHS option as shown in the following example.
engine = aw.reporting.ReportingEngine()
engine.options |= aw.reporting.ReportBuildOptions.REMOVE_EMPTY_PARAGRAPHS
engine.build_report(...)
The difference in the engine’s behavior when the option is applied and not applied is illustrated by the following examples.
Example 1
Template document
Prefix
<<[""]>>
Suffix
Result document without ReportBuildOptions.REMOVE_EMPTY_PARAGRAPHS applied
Prefix
Suffix
Result document with ReportBuildOptions.REMOVE_EMPTY_PARAGRAPHS applied
Prefix
Suffix
Example 2
Template document
Prefix
<<if [false]>>
Text to be removed
<</if>>
Suffix
Result document without ReportBuildOptions.REMOVE_EMPTY_PARAGRAPHS applied
Prefix
Suffix
Result document with ReportBuildOptions.REMOVE_EMPTY_PARAGRAPHS applied
Prefix
Suffix
Example 3
Note – In this example, persons
is assumed to be a data table having a field Name
.
Template document
Prefix
<<foreach [in persons]>>
<<[Name]>>
<</foreach>>
Suffix
Result document without ReportBuildOptions.REMOVE_EMPTY_PARAGRAPHS applied
Prefix
John Doe
Jane Doe
John Smith
Suffix
Result document with ReportBuildOptions.REMOVE_EMPTY_PARAGRAPHS applied
Prefix
John Doe
Jane Doe
John Smith
Suffix
Accessing Missing Members of Data Objects
By default, LINQ Reporting Engine forbids access to missing members of data objects used to build a report in template expressions. On attempt to use a missing member of a data object, the engine throws an exception then.
But in some scenarios, members of data objects are not exactly known while designing a template.
To support such scenarios, the engine provides an option to treat missing members of data objects as null literals. You can enable the option as shown in the following example.
engine = aw.reporting.ReportingEngine()
engine.options |= aw.reporting.ReportBuildOptions.ALLOW_MISSING_MEMBERS
engine.build_report(...)
Consider the following example. Given that r
is a data item instance that does not have a field Missing
, by default, the following template expression causes the engine to throw an exception while building a report.
<<[r.Missing]>>
However, if ReportBuildOptions.ALLOW_MISSING_MEMBERS is applied, the engine treats access to such a field as a null literal, so no exception is thrown and simply no value is written to the report then.
Inlining Syntax Error Messages into Templates
By default, LINQ Reporting Engine throws an exception when encounters a template syntax error. Such an exception provides information on a reason of the error and specifies a tag or expression part where the error is encountered. In most cases, this information is enough to find a place in a template causing the error and fix it.
However, when dealing with complex templates containing a large number of tags, it becomes harder to find an exact place in a template causing an error. To make things easier, the engine supports the ReportBuildOptions.INLINE_ERROR_MESSAGES option that enables inlining of a syntax error message into a template document at an exact position where the error occurs during runtime.
Note – A template syntax error message is written using a bold font to make it more apparent.
Consider the following template.
<<var [name]>>
By default, such a template causes the engine to throw an exception while building a report. However, when ReportBuildOptions.INLINE_ERROR_MESSAGES is applied, no exception is thrown and the report looks as follows then.
<<var [name] Error! An assignment operator is expected. >>
Note – Only messages describing errors in template syntax can be inlined; messages describing errors encountered during expressions’ evaluation cannot.
When ReportBuildOptions.INLINE_ERROR_MESSAGES is applied, a Boolean
value returned by a ReportingEngine.build_report overload indicates whether building of a report was finished successfully or was interrupted because of a template syntax error. This enables you to process reports which building succeeded or failed differently as shown in the following code snippet.
engine = aw.reporting.ReportingEngine()
engine.options |= aw.reporting.ReportBuildOptions.INLINE_ERROR_MESSAGES
if (engine.build_report(...)) :
# Do something with a successfully built report.
else :
# Do something with a report containing a template syntax error.
Note – When ReportBuildOptions.INLINE_ERROR_MESSAGES is not applied, ReportingEngine.build_report overloads return True
if there were no template syntax errors encountered or throw an exception otherwise.
Optimizing Reflection Calls
LINQ Reporting Engine uses reflection calls while accessing members of custom external types. However, reflection calls are much slower than direct calls, which create a performance overhead.
That is why, the engine provides a strategy minimizing the reflection usage. The strategy is based upon the runtime type generation. That is, the engine generates a proxy type per an external type. The proxy directly calls members of the corresponding external type, the engine to access these members in a uniform way with no reflection involved. The proxy is lazily initialized and reused in further. Thus, the reflection is used only while building the proxy.
Although this strategy can significantly minimize the reflection usage in a long run, it creates a performance overhead of the runtime type generation. So, if you deal with small data collections all the time while building your reports, consider the disabling of the strategy. You can control the enabling of the strategy through the ReportingEngine.use_reflection_optimization static property. By default, the strategy is enabled.