Syntax Rules and Policy Check Examples
This page provides an overview on syntax rules, common check types and useful best practices for creating policy content.
This page provides an overview on syntax rules, common check types and useful best practices for creating policy content.
The validation engine of Configuration and Security Analytics is running directly on SAP HANA database level. It is basically using HANA SQL as described in the SAP HANA SQL and System Views Reference. However, not all features are supported. The actual SQL used for validations is generated automatically based on the XML coding of a policy.
Policies are written in an xml format. Out of this the SQL coding for the validation is created automatically when the policy is generated. This section provides an overview on the syntax elements that are required in every policy.
Each policy starts with an element containing the attributes related to the used xml version and encoding. When creating a new policy, this element is created automatically. It must not be changed.
<?xml version="1.0" encoding="utf-8"?>
All other elements of a policy are opened and closed. Elements are specified in <>. The closing element starts with </.
<targetsystem> </targetsystem>
A policy must contain at least the following elements:
<targetsystem>
<configstore>
<checkitem>
<compliant>
</compliant>
<noncompliant>
</noncompliant>
</checkitem>
</configstore>
</targetsystem>
Some of these mandatory elements do not necessarily need to contain content, e.g the noncompliant rule can remain empty in some cases. In this case the syntax can be shortened:
<noncompliant/> instead of: <noncompliant> </noncompliant>
In addition the following elements can be used:
<complianttext> <complianttext/>
<noncomplianttext> <noncomplianttext/>
<joinstore> <joinstore/>
<joincomplianttext> </joincomplianttext>
<joinnoncomplianttext> </joinnoncomplianttext>
You can also add comments to the policy:
<!-- -->
While you can learn many syntax elements by starting your work directly with the use of policies published on the SAP Github, additional documentation may be useful in some cases. Each of the elements given above may contain additional attributes. These are explained on the next sections of this page. After this we provide documentation on typical syntax for check items. At the end we will provide syntax examples from the perspective of typical validation needs (driven by semantics rather than by language).
Rule example: Password length is compliant if the parameter value is greater equal 8
<configstore name="ABAP_INSTANCE_PAHI">
<checkitem desc="First item check " id="1.0.0.0">
<compliant>name= 'login/min_password_lng' and value>= 8</compliant>
<noncompliant>name= 'login/min_password_lng' and not ( value >= 8 )</noncompliant>
</checkitem>
</configstore>
Note:
The following is an example of a HANA parameter check (from now on we omit the element about the target system):
<configstore name="HDB_PARAMETER">
<checkitem desc="1.1. sslCryptoProvider" id="001">
<compliant>FILE_NAME = 'global.ini' and SECT = 'communication' and NAME = 'sslcryptoprovider' and
VALUE = 'commoncrypto' and LAYER like '%' and HOST like '%' and TENANT like '%' </compliant>
<noncompliant>FILE_NAME = 'global.ini' and SECT = 'communication' and NAME = 'sslcryptoprovider' and
NOT VALUE = 'commoncrypto' and LAYER like '%' and HOST like '%' and TENANT like '%' </noncompliant>
</checkitem>
<checkitem desc="1.1. sslMinProtocolVersion" id="002">
<compliant>FILE_NAME = 'global.ini' and SECT = 'communication' and NAME = 'sslminprotocolversion' and
VALUE = 'TLS12' and LAYER like '%' and HOST like '%' and TENANT like '%' </compliant>
<noncompliant>FILE_NAME = 'global.ini' and SECT = 'communication' and NAME = 'sslminprotocolversion' and
NOT VALUE = 'TLS12' and LAYER like '%' and HOST like '%' and TENANT like '%' </noncompliant>
</checkitem>
</configstore>
Figure 1: Validation Result of 2 check examples
Compliant or non-compliant rules are defined as a subset of SQL WHERE clause. The following syntax could be used:
Note:
The following section gives you examples for the use of basic syntax elements. Focus is on checking simple numerical values.
You can combine predicates using and, or, not and brackets.
<checkitem desc="3.1.1.1 System configuration (profile parameters)" id="001">
<compliant>NAME = 'login/failed_user_auto_unlock' and VALUE = '0'</compliant>
<noncompliant>NAME = 'login/failed_user_auto_unlock' and NOT VALUE = '0' </noncompliant>
</checkitem>
<checkitem desc="3.1.1.5. Logging (profile parameters)" id="002">
<compliant>NAME = 'rec/client' and VALUE = 'ALL' </compliant>
<noncompliant>NAME = 'rec/client' and NOT ( VALUE = 'ALL' ) </noncompliant>
</checkitem>
<checkitem desc="3.1.1.1. login/min_password_lng" id="003">
<compliant>name= 'login/min_password_lng' and value > 8</compliant>
<noncompliant>name= 'login/min_password_lng' and not ( value > 8 )</noncompliant>
</checkitem>
<checkitem desc="3.1.1.1. login/password_max_idle_initial" id="004">
<compliant>name= 'login/password_max_idle_initial' and value < 14</compliant>
<noncompliant>name= 'login/password_max_idle_initial' and not ( value < 8 )</noncompliant>
</checkitem>
<configstore name="HDB_SYSTEM_USER">
<checkitem desc="System user valid_from lt last pw change time" id="DATE_COMP">
<compliant>USER_NAME = 'SYSTEM' and CsaToBigInt(VALID_FROM) < CsaToBigInt(LAST_PASSW_CHANGE_TIME)</compliant>
<noncompliant>USER_NAME = 'SYSTEM' and not CsaToBigInt(VALID_FROM) < CsaToBigInt(LAST_PASSW_CHANGE_TIME)</noncompliant>
</checkitem>
</configstore>
<checkitem desc="System configuration (profile parameters)" id="005">
<compliant>NAME = 'login/min_password_digits' and VALUE >= 1 </compliant>
<noncompliant>NAME = 'login/min_password_digits' and NOT (VALUE >= 1) </noncompliant>
</checkitem>
<checkitem desc="3.1.1.1. login/password_max_idle_initial" id="006">
<compliant>name= 'login/password_max_idle_initial' and value<= 14</compliant>
<noncompliant>name= 'login/password_max_idle_initial' and not ( value <= 8 )</noncompliant>
</checkitem>
<checkitem desc="3.2.3. DDIC is LOCKED" id="007">
<compliant>CLIENT like '%' and USERNAME = 'DDIC' and ( LOCKED = 'X' or EXISTING <> 'X' )
</compliant>
<noncompliant>CLIENT like '%' and USERNAME = 'DDIC' and not ( LOCKED = 'X' or EXISTING <> 'X' )
</noncompliant>
</checkitem>
<checkitem desc="System configuration (profile parameters)" id="009">
<compliant>
NAME = 'login/fails_to_session_end' and VALUE between 1 and 3
</compliant>
<noncompliant>
NAME = 'login/fails_to_session_end' and not (VALUE between 1 and 3)
</noncompliant>
</checkitem>
VALUE in ( 'A', 'b' )
Performs a comparison to see if a character string matches a specified pattern.
Wildcard characters ( % ) and ( _ ) can be used in <pattern_expression>. The percentage sign (%) wildcard matches zero or more characters. The underscore (_) wildcard matches exactly one character.
To match a percentage sign or underscore with the like predicate, an escape character must be placed in front of the wildcard character. Use escape <escape_expression> to specify the escape character you are using. For example, like 'data_%' escape '_' matches the string data%, and like 'data__%' escape '_' (that is, two underscores, followed by a percent sign) matches a string that starts with 'data_'.
Examples:
VALUE like '%'
TEXT like '%P TP=*%'
Performs regular expression matching.
like_regexpr '.*TP=[^\*].*'
<checkitem desc="3.2.3. mandatory System Replication" id="008">
<compliant>FILE_NAME= 'global.ini' and LAYER like '%' and HOST like '%' and TENANT like '%' and
SECT = 'system_replication' and NAME = 'site_id' and length(VALUE) > 0 </compliant>
<noncompliant>FILE_NAME = 'global.ini' and LAYER like '%' and HOST like '%' and TENANT like '%' and
SECT = 'system_replication' and NAME = 'site_id' and not length(VALUE) > 0 </noncompliant>
</checkitem>
This section demonstrates some use cases of advanced syntax elements.
<checkitem desc="System configuration (profile parameters) - secret" id="00015" system_roles="PROD">
<compliant>NAME = 'login/password_max_idle_productive' and VALUE between 1 and 90 </compliant>
<noncompliant>NAME = 'login/password_max_idle_productive' and NOT (VALUE between 1 and 90) </noncompliant>
</checkitem>
Additive operators ( +,- ) and multiplicative operators ( *, / ) can be used in checks using function CsaToBigInt. The function can be applied to any numerical field of a configuration item (e.g. “VALUE” of a parameter or a field containing timestamp information)
The option to apply the value of a second item for calculation is realized by using join conditions. The join condition must restrict the items as good as possible, best one item. The value is then used as s1.VALUE
The given examples are supposed to help to understand the functionality. They are not meant as recommendations.
This example check whether ‘pythontrace' 'filesize_limit' is 1000 times as big as 'executed_statement' ‘maxfilesize'.
<configstore name="HDB_PARAMETER">
<checkitem desc=" filesize_limit and maxfilesize" id="HDBPARA_CFUNC3">
<compliant>FILE_NAME = 'attributes.ini' and SECT = 'pythontrace' and NAME like 'filesize_limit' and (CsaToBigInt(s0.VALUE) >= CsaToBigInt(s1.VALUE)*1000)</compliant>
<noncompliant>FILE_NAME = 'attributes.ini' and SECT = 'pythontrace' and NAME like 'filesize_limit' and not (CsaToBigInt(s0.VALUE) >= CsaToBigInt(s1.VALUE)*1000)</noncompliant>
<joinstore name="HDB_PARAMETER">
<joincompliant>FILE_NAME = 'global.ini' and SECT = 'executed_statement' and NAME = 'maxfilesize' and CsaToBigInt(VALUE) >= 0</joincompliant>
<joinnoncompliant>FILE_NAME = 'global.ini' and SECT = 'executed_statement' and NAME = 'maxfilesize' and CsaToBigInt(VALUE) >= 0</joinnoncompliant>
</joinstore>
</checkitem>
</configstore>
The following example compares the 'plan_cache_size' with the sum of 'global_allocation_limit' and 'TotalVisibleMemorySize' divided by 0.5. The first join is referred as s1, the second as s2.
The check value itself is referred as s0.
<configstore name="HDB_PARAMETER">
<checkitem desc="Plan Cache Size, Global Allocation Limit MB, and Host Memory" id="PlanLimitMemory">
<compliant>FILE_NAME = 'indexserver.ini' and LAYER like '%' and NAME = 'plan_cache_size' and CsaToBigInt(s0.value) >= round((CsaToBigInt(s1.VALUE) + CsaToBigInt(s2.VALUE))/0.5)</compliant>
<noncompliant>FILE_NAME = 'indexserver.ini' and LAYER like '%' and NAME = 'plan_cache_size' and CsaToBigInt(s0.VALUE) < round((CsaToBigInt(s1.VALUE) + CsaToBigInt(s2.VALUE))/0.5)</noncompliant>
<joinstore name="HDB_PARAMETER">
<joincompliant>FILE_NAME = 'global.ini' and LAYER like '%' and NAME = 'global_allocation_limit'</joincompliant>
<joinnoncompliant>FILE_NAME = 'global.ini' and LAYER like '%' and NAME = 'global_allocation_limit'</joinnoncompliant>
</joinstore>
<joinstore name="SAP_ITSAMOperatingSystem">
<joincompliant>NAME = 'TotalVisibleMemorySize'</joincompliant>
<joinnoncompliant>NAME = 'TotalVisibleMemorySize'</joinnoncompliant>
</joinstore>
</checkitem>
</configstore>
The parameter 'max_gc_parallelity' is supposed to have a value between CPUCount divided by 2 and 72.
Remark: CPUCount stands for number of CPU threads.
<configstore name="HDB_PARAMETER">
<checkitem desc="max_gc_parallelity CPU threads" id="CPUCNT">
<compliant>FILE_NAME = 'global.ini' and SECT = 'persistence' and NAME = 'max_gc_parallelity' and (CsaToBigInt(VALUE) between (CsaToBigInt(s1.VALUE) * 0.5) and 72) </compliant>
<noncompliant>FILE_NAME = 'global.ini' and SECT = 'persistence' and NAME = 'max_gc_parallelity' and not (CsaToBigInt(VALUE) between (CsaToBigInt(s1.VALUE) * 0.5) and 72)</noncompliant>
<joinstore name="SAP_ITSAMComputerSystem">
<joincompliant>AREA = 'SAP_ITSAMComputerSystem' and NAME = 'CPUCount'</joincompliant>
<joinnoncompliant>AREA = 'SAP_ITSAMComputerSystem' and NAME = 'CPUCount'</joinnoncompliant>
</joinstore>
</checkitem>
</configstore>
Limitations:
- Maximum value: 2^63 – 1 (BigInt)
- Aggregate functions like SUM, COUNT, AVG, MIN, MAX are not supported
- The function CsaToBigInt is available starting with FRUN 3.0 FP01. Before use instead "to_bigint(coalesce(substr_regexpr('^([[:blank:]]{0,})(\d{1,19})([[:blank:]]{0,})$' in VALUE group 2), '-1' ))"
Each policy contains exactly one element targetsystem. Tag <targetsystem> is placed in the line directly after the specification of xml version and encoding. Tag </targetsystem> is the corresponding closing bracket. It is placed at the end of the policy.
Attribute "desc" contains the targetsystem description. Each policy must contain exactly one targetsystem description.
Attribute "id" contains the target system definition Id. Each policy must contain exactly one target system id.
Each policy contains at least one configstore. It is mandatory to specify the name of the config store through the attribute name. All other attributes to this element are optional and only used in exceptional cases.
Tags <configstore> and </configstore> are the opening and closing bracket for a configstore.
Attribute "name" specifies the technical name of the config store. The attribute is mandatory.
Syntax example: <configstore name = "COMP_LEVEL">
Attribute "name_extended" can be used as further attribute in case the technical name of a config store is not unique (optional).
Possible values of attribute "name_extended" can be identified with view CCDB_DIRV.
The attribute "sci_id" must be used in case the technical name of a config_store is not unique (mandatory).
Attribute "system_type" can be set additional to restrict the system type (e.g. dir.system="JAVA").
This may only be necessary if the same config store is used cross several system types (currently the case for COMP_LEVEL which contains the software components for ABAP and JAVA systems) (optional).
In addition to the attributes of element configstore you can use the following attributes for joinstores. When building a join condition for a check item, the element joinstore allows to implement a join against configuration items from any config store.
Tags <joinstore> and </joinstore> are the opening and closing bracket for a check.
The elements <joincompliant> and <joinnoncompliant> work the same way as <compliant> and <noncompliant> for a simple check item. They contain the technical rules that the configuration item is checked against.
The elements <joincomplianttext> and <joinnoncomplianttext> work the same way as the elements <complianttext> and <noncomplianttext> for a simple check item. They are used to replace the technical rule with a text describing the rule elements. If used those texts replace the SQL rules in the corresponding fields of the UI (optional).
Attribute "minus" applies only to element "joinstore". It controls the way the non-compliant join is transferred into SQL:
Attribute "no_data_found" helps to handle join conditions that do not apply to any configuration item.
The standard behaviour of the validation is that each check generates an SQL statement which returns the status “item not found” in case both statements for the compliant and non-compliant check do not return any results for a system. However, in some cases one of the join statements introduces a validity constraint. In such a case, it is possible to add this validity constraint also to the SQL statement checking “item not found”.
no_data_found="Yes"
Best Practice:
Example 1:
Value of HANA parameter: FILE_NAME = 'global.ini', SECT = 'parallel', and NAME = 'num_cores' should be checked, however, only for HANA systems running a version between 12202 and 12206. To avoid “item not found” results for HANA systems running on a version not between 12202 and 12206 it is necessary to add an attribute to this join to mark it to be used also in the “item not found”-SQL to include only those which are running the correct HANA version (optional).
Use of no_data_found="Yes" operator (Note: The operator can be applied in a join condition or in a check item)
Output of a "data not found" record when no_data_found="Yes" is not set.
In some cases the columns of the joined config store needs to be suppressed to avoid getting multiple tuples of the join result. This behaviour can be achieved with attribute "not_in_output".
Supported as of FRUN 2.00 SP00
Checkitems are the element that contain the actual validation rules for determining compliance and noncompliance of a configuration. A policy can contain multiple checkitems. Checkitems should be grouped by config store. Tags <checkitem> and </checkitem> are the opening and closing bracket for a check.
Various attributes can be used to control the behaviour of check items in a validation run. This section provides details and some examples.
Attribute "desc" contains a description of the check item. Between " " you can enter any content that helps understanding the meaning of the check.
Attribute "id" contains the check Id of the check item. In most cases it will refer to an identifier that is used in your hardening guide or security procedure.
It is not required to have a numeric id. You can also use characters or special characters.
not_found="ignore"
Available since FRUN 3.00 FP00 together with 'Composite Note for Advanced Configuration Monitoring'.
Effect:
not_found="negative"
Attribute "not_found" controls the compliance status of check item in case its rules does not select any config item.
Effect:
not_found="positive"
Effect:
Use of the not_found="positive" operator for a check item
With not_found="positive" a check item is rated compliant when not matching to compliant/noncompliant rules (default: "negative")
The attribute "operator" allows to apply certain predefined operators to the check item. One example are the operators "EXIST" and "NOT_EXIST".
Effect:
For checking obsolete or forbidden parameters you can use the operator "NOT_EXIST" by adding the separator “:” and the name of the attribute.
Note: " Operators "EXIST and "NOT_EXIST" do not support joins. The attribute "system_roles" is applied as join and is not accepted in combination with the operators "EXIST and "NOT_EXIST".
Coming with FRUN 5.0 the attribute "system_attributes" is accepted.
<configstore name="ABAP_INSTANCE_PAHI">
<checkitem desc="3.4 Password parameter does not exist" id="00003" operator="EXIST">
<compliant>
NAME = 'login/min_password_lng'
</compliant>
<noncompliant>
NAME = 'login/min_password_lng'
</noncompliant>
</checkitem>
</configstore>
<configstore name="SICF_SERVICES">
<checkitem desc="obsolete" id="FormToRfc" operator="not_exist:URL">
<compliant> ICF_NAME like '%' and URL = '/sap/bc/FormToRfc/'</compliant>
<complianttext/>
<noncompliant>ICF_NAME like '%' and URL = '/sap/bc/FormToRfc/'</noncompliant>
<noncomplianttext/>
</checkitem>
</configstore>
Syntax for using the "EXIST" operator
Validation output with the "EXIST" operator
The attribute "operator" allows to apply certain predefined operators to the check item.
For ABAP notes checks the "operator" attribute can be set to “check_note” or “check_note:<10 digits note number>”.
Attribute "system_roles" can be used to pin the check item to a specific system role (or IT admin roles).
Attribute system_attributes was introduced to pin a check item to a specific system attribute, but its functionality has been enhanced within the feature packs as explained below.
In general, it allows to pin a check to any of the parameters that are reported by the report Report COF_CIM_TREE_TS (***) and are flagged with BELONGS_TO_RT = X.
Example: system_attributes="DB_TENANT:!TRUE"
Example: system_attributes="DB_REPL_OP_MODE:!LOGREPLAY"
Available functionality depending on SAP Focused Run Feature Package:
Up to SAP Focused Run 2.00 FP02
Attribute system_attributes filters with regard to attributes that are associated to the type of landscape object that corresponds to the given store, e.g.: Store ABAP_INSTANCE_PAHI is implemented on instance level. Therefore, it is not possible to apply the Extended Sid as system attribute (system_attributes="ESID:<ExtendedSid>".
Since SAP Focused Run 2.00 FP03
The LMDB custom attributes are available as landscape parameters with the following naming convention "_LA_<Custom attribute name>". In opposite to regular parameters they always refer to the root object which is the managed system. Example: Even if the check belongs to a store that is attached to an instance, the "_LA_<Custom attribute name>" parameter is evaluated at the root object which is the managed system.
Since SAP Focused Run 3.00 FP00
By default, all parameters refer to the root object which is the managed system. For example, this means that it is now possible to use system_attributes="ESID:<ExtendedSid> also for stores which are not attached to the managed system, but to an abap instance.
To overrule this behaviour, the prefix "-" can be inserted in front of the parameter name. As a result, the parameter of the landscape object the store is attached to is used to determine the parameter value. Example: system_attributes="-CLIENT:000" can be used to restrict a check which is based on a store attached to an ABAP client to the client '000'.
Compatibility exemption: The parameter name "CLIENT" is treated like "-CLIENT".
Report COF_CIM_TREE_TS (***)
Input fields of Report COF_CIM_TREE_TS can be taken from the LMDB Object Maintenance:
The report outputs the landscape hierarchy which is used by the Advanced Configuration Monitoring use case.