DICOM audit support
The ipf-commons-audit
module contains a reimplementation and improvement of the ATNA functionality
contained in the ipf-oht-atna-*
modules that originate from the Eclipse OpenHealthTools project.
This includes building DICOM/ATNA-compliant Audit Records as well as queueing and sending them to
an Audit Repository.
The most important improvements are:
- type-safe Audit Event builders, that enforce setting mandatory members and avoid setting
coded values into the wrong places (e.g. accidentally setting a
ParticipantObjectTypeCodeRole
code asParticipantObjectTypeCode
code). - possibility to select an Audit Message XML schema that belongs to a specific DICOM version
- possibility to plug-in your own exception handler in case the Audit Repository is not reachable
- configuration is done via an
AuditContext
bean instead of a static global class. You can create and use as manyAuditContext
configuration beans as you wish.
Constructing an Audit Message
All of the IHE components implemented by IPF automatically use the audit support of this module for ATNA auditing. You only need to configure an AuditContext bean. When you use one of the Spring Boot starter modules, this bean is already provided for you.
In order to create and submit your own Audit Message, perform the following steps
- Construct your Audit Message by using one of the builder classes contained in the
org.openehealth.ipf.commons.audit.event
package. Note that the IHE modules (such asipf-commons-ihe-core
oripf-commons-ihe-xds
) inherit from some of these builders to create IHE-style ATNA-compliant Audit messages more easily. - Call the
getMessage
orgetMessages
method to obtain theAuditMessage
from the builder. - Submit the Audit Message to the configured destination by calling
AuditContext#audit(AuditMessage...)
method.
Every Audit Message and Audit Message builder also has a validate
method that check basic
restrictions defined by DICOM or IHE.
The AuditMessageBuilder
implementations are modelled corresponding to the definitions of the
DICOM Specific Audit Messages. The delegate builder class IHEAuditMessageBuilder
has builder sub classes that correspond with the IHE ITI specification, Volume 2a, section 3.20.4.1.
The DICOM serialization strategies produce XML files that validate against the schema of the respective revision of the DICOM Audit Message Schema.
For more details on the API, please study the javadocs.
Example
Auditing an application start event
auditContext.audit(
new ApplicationActivityBuilder.ApplicationStart(EventOutcomeIndicator.Success)
.setAuditSource(auditContext)
.setApplicationParticipant(
appName,
null,
appName,
AuditUtils.getLocalHostName())
.addApplicationStarterParticipant(System.getProperty("user.name"))
.getMessage()
);
Auditing a change to a user account:
auditContext.audit(
new SecurityAlertBuilder(EventOutcomeIndicator.Success, null, EventTypeCode.UserSecurityAttributesChanged)
.addActiveParticipant(adminUserId, null, adminUserName, true, null, networkId)
.addParticipantObjectIdentification(
ParticipantObjectIdTypeCode.UserIdentifier,
targetUserName,
null,
null,
targetUserId,
ParticipantObjectTypeCode.Person,
ParticipantObjectTypeCodeRole.User,
null, null)
.getMessage()
);
Hints
The org.openehealth.ipf.commons.audit.utils.AuditUtils
class contains static methods to obtain
runtime information like process ID, current user, local host and IP address.
Configuration of the AuditContext
The AuditContext
interface (and its DefaultAuditContext implementation)
is the only place to configure static details for auditing, e.g. whether auditing is activated, the location of the Audit Repository, or
the transmission protocol. It also allows to setup strategies for serialization, whether to send synchronously or asynchronously, and how errors are handled.
Generic properties
Property | Default | Description |
---|---|---|
auditEnabled |
false | Whether audit is sent to the repository or not |
auditRepositoryHost |
localhost | Host name of the audit repository where audit records are sent to |
auditRepositoryPort |
514 | Port of the the audit repository where audit records are sent to |
auditRepositoryTransport |
UDP | Transport protocol. See info about Audit Transmission Protocols below. |
Content properties
Property | Default | Description |
---|---|---|
sendingApplication |
IPF | Sending application for the Syslog header info |
auditSourceId |
IPF | Audit source ID for the source identification of the audit message |
auditEnterpriseSiteId |
IPF | Audit enterprise site ID for the source identification of the audit message |
auditSource |
9 (Other) | Audit source type for the source identification of the audit message |
includeParticipantsFromResponse |
false | Whether to include participant objects from a response into the audit message |
Advanced properties
Property | Default | Description |
---|---|---|
auditTransmissionProtocol |
Instance of UDPSyslogSenderImpl |
Transport implementation. Overrules auditRepositoryTransport |
auditMessageQueue |
Instance of SynchronousAuditMessageQueue |
Audit message dispatcher implementation |
serializationStrategy |
Instance of Current (i.e. DICOM2017c) |
Serialization implementation |
tlsParameters (3.7) |
using system defaults | Parameters used for TLS-based transmission |
auditMetadataProvider (3.7) |
Instance of DefaultAuditMetadataProvider | Provider for header data used for SYSLOG-based transmission |
auditMessagePostProcessor |
no-op | Audit Message Postprocessing, called before audit message is dispatched |
auditExceptionHandler |
Instance of LoggingAuditExceptionHandler |
Handler to be called if the delivery of audit message to the audit repository has failed |
auditValueIfMissing |
UNKNOWN |
Value used if a mandatory audit attribute is not present |
wsAuditDatasetEnricher |
no-op | Audit dataset enricher for Web Service based transactions (IPF 5.0+) |
fhirAuditDatasetEnricher |
no-op | Audit dataset enricher for FHIR based transactions (IPF 5.0+) |
contextRegistry |
default context registry with Slf4J propagation | Micrometer Context Registry used for propagating thread-local variables into the sending thread when using an AsynchronousAuditMessageQueue |
The default setup is to send Audit Messages via UDP to localhost:514
, and handle delivery errors by just logging them.
For production usage, it is usually required to configure:
- a TLS connection to a remote Audit Record Repository
- some decent strategy for handling failed connections
Audit Transmission Protocols
The transmission protocol determines the network protocol used for sending an audit record to the Audit Record Repository.
auditRepositoryTransport |
auditTransmissionProtocol class |
Description |
---|---|---|
UDP |
UDPSyslogSenderImpl | UDP transport as SYSLOG record without delivery guarantee. Failed delivery is ignored. |
TLS |
TLSSyslogSenderImpl | Blocking TLS transport as SYSLOG record |
NIO-TLS or NETTY-TLS |
NettyTLSSyslogSenderImpl | Non-blocking TLS transport as SYSLOG record. Requires Netty library on the classpath |
REACTOR-NETTY-TLS |
ReactorNettyTLSSyslogSenderImpl | Reactive TLS transport as SYSLOG record. Requires Reactor-Netty library on the classpath |
FHIR-REST-TLS |
ApacheFhirRestTLSAuditRecordSender | as of IPF 4.8.0: Blocking HTTPS transport as FHIR R4 AuditEvent resource |
FHIR-REST-APACHE5-TLS |
Apache5FhirRestTLSAuditRecordSender | as of IPF 5.0.0: Blocking HTTPS transport as FHIR R4 AuditEvent resource |
Audit Message Queues
There are a couple of implementations for “how” IPF handles audit records. Some implementations circumvent the Audit Transmission Protocol.
auditMessageQueue class |
Description |
---|---|
SynchronousAuditMessageQueue | Synchronously pass the audit record to the auditTransmissionProtocol instance |
AsynchronousAuditMessageQueue | Asynchronously pass the audit record to the auditTransmissionProtocol instance. Must be initialized with an ExecutorService |
JMSAuditMessageQueue | Send the audit record to a JMS queue. SYSLOG header data is sent as JMS properties. Requires JMS API library. |
BasicHttpAuditMessageQueue (3.7) | Send the audit record to a HTTP service. SYSLOG header data is sent as HTTP properties. |
LoggingAuditMessageQueue | Just log the audit record to an SLF4J logger |
CamelAuditMessageQueue | Send the audit record via a Camel producer. SYSLOG header data is sent as Camel headers. |
RestTemplateAuditMessageQueue | Send the (FHIR) audit record via a Spring RestTemplate instance. SYSLOG header data is sent as HTTP properties. |
RestClientAuditMessageQueue | Send the (FHIR) audit record via a Spring RestClient instance. SYSLOG header data is sent as HTTP properties. |
CompositeAuditMessageQueue | Send the audit record sequentially using one of the implementations listed above |
Audit Dataset Enrichers (5.0)
The following enrichers are provided out of the box.
Enrichers for Web Service based transactions (implementations of WsAuditDatasetEnricher):
Audit dataset enricher class | Description |
---|---|
XuaWsAuditDatasetEnricher | Fulfills requirements of the IHE XUA profile |
SwissEprWsAuditDatasetEnricher | Fulfills both IHE XUA requirements and the requirements of the Swiss Electronic Patient Record |
Enrichers for FHIR based transactions (implementations of FhirAuditDatasetEnricher):
Audit dataset enricher class | Description |
---|---|
SwissEprFhirAuditDatasetEnricher | Fulfills the requirements of the Swiss Electronic Patient Record |
TLS Parameters (3.7)
With TLS-based audit transmission, the default TLS parameters (as passed using the javax.net.ssl.*
system properties)
are used. However, sometimes the TLS Audit connection requires a different TLS configuration. In this case, an instance
of CustomTlsParameters can be used to specifically assign a different TLS setup
to the AuditContext
.
TLS/UDP Syslog Server (4.0)
As of IPF 4.0, the ipf-commons-audit
module comes with a server-side component that implements a
Syslog server for both the RFC 5425 (TLS transport) and RFC 5426 (UDP transport) specifications.
These servers are primarily meant for integration testing of functionality that emits ATNA audit records, but can also be used as basis for standalone applications. Note that only the Syslog endpoints and some parsers are provided, however, the library doesn’t include a persistence layer or any other production-level features.
TLS (TlsSyslogServer) and UDP (UdpSyslogServer) servers are used very similarly, so the following examples primarily cover the TLS server.
Required dependencies
Two optional Maven dependencies need to be included (versions can be imported from the dependencies
module):
<dependency>
<groupId>io.projectreactor.netty</groupId>
<artifactId>reactor-netty-core</artifactId>
</dependency>
<dependency>
<groupId>com.github.palindromicity</groupId>
<artifactId>simple-syslog-5424</artifactId>
</dependency>
Instantiation and Start
The Syslog servers parses an incoming syslog record into a Map<String, Object>
of its attributes.
A SyslogServer instance is initialized with
- a regular consumer the processes this map
- an error consumer that processes any Exception that may have occurred during parsing
- an instance of
TlsParameters
and is start
ed with
- a host (
localhost
for local access,0.0.0.0
for external access) - a port (the RFCs define 6514 for TLS and 514 for UDP)
The number of connections per remote host is not restricted.
The regular consumer defines what happens with the syslog record. We provide one consumer
(org.openehealth.ipf.commons.audit.server.support.SyslogEventCollector
) that simply collects
the records in memory and can be configured with some expectations for unit tests. See the
following snippet for a potential unit test:
var defaultTls = TlsParameters.getDefault();
auditContext.setTlsParameters(defaultTls);
auditContext.setAuditRepositoryTransport("TLS");
var count = 100;
var consumer = SyslogEventCollector.newInstance()
.withExpectation(count) // expecting count records to arrive
.withDelay(100); // artificial delay of 100ms for handling a record
try (var ignored = new TlsSyslogServer(consumer, Throwable::printStackTrace, defaultTls)
.start("localhost", port)) // For a real server you would have to block here.
{
IntStream.range(0, count).forEach(i -> sendAudit()); // send out "count" records via AuditContext#audit
assertTrue(consumer.await(5, TimeUnit.SECONDS)); // wait max. 5 seconds for all records to arrive
} // SyslogServers are AutoCloseable!
The consumer executes asynchronously using a variable number of threads (depending on producer/consumer load) rather than in the receiver thread, so that the test finishes within 5 seconds even though consuming sleeps for 100ms for each of the 100 messages.
ATNA unmarshaller
IPF also provides an unmarshaller (org.openehealth.ipf.commons.audit.unmarshal.dicom.DICOMAuditParser
) that
allows to parse syslog XML payload following the current DICOM specification (used by the IHE ATNA profile)
back into an instance of org.openehealth.ipf.commons.audit.model.AuditMessage
while optionally validating
the XML content against corresponding the XML schema.
This would allow to write an ATNA Consumer for the Syslog Server that extracts structured data from the ATNA payload, e.g. for offering detailed queries to retrieve specific records.
Beta disclaimer
The Syslog server is a new feature and, while being stable enough for “happy case” usage, it may still require improvement and configuration possibilities, particularly to handle edge cases. You are invited to give it a test drive and report and findings or suggestions. Thank you!