Karnak
Karnak is a DICOM gateway for data de-identification and DICOM attribute normalization.
Karnak manages a continuous DICOM flow with a DICOM listener as input and a DICOM and/or DICOMWeb as output.
Karnak is a DICOM gateway for data de-identification and DICOM attribute normalization.
Karnak manages a continuous DICOM flow with a DICOM listener as input and a DICOM and/or DICOMWeb as output.
Minimum docker version required: 20.10
generateSecrets.sh
to generate the secrets required by KarnakCommands from the root of this repository.
docker compose pull
docker compose up -d
docker compose down
docker compose down -v
docker compose logs -f
You can generate the secrets with the generateSecrets.sh
script available at the root of the karnak-docker repository (adapt the script to your system if necessary).
Note: These following secrets are stored in files and use the environment variables ending with _FILE (see ‘Environment variables’ below).
Before starting docker-compose make sure that the secrets folder and the following secrets exist:
karnak_login_password
karnak_postgres_password
Configure the logging system in Karnak
Create a systemd service for the Karnak DICOM gateway
Karnak offers the possibility of injecting your own log configuration.
The logging system used by Karnak is logback. For more details on logback, it provides a manual
Some variables are available and can be used in the logback file.
issuerOfPatientID
are the issuer of patient ID before the deidentificationPatientID
are the patient ID before the deidentificationSOPInstanceUID
are the SOPInstanceUID before the deidentificationDeidentifySOPInstanceUID
are the SOPInstanceUID after the deidentificationSeriesInstanceUID
are the SeriesInstanceUID before the deidentificationDeidentifySeriesInstanceUID
are the SeriesInstanceUID after the deidentificationProjectName
are the project name used for the deidentificationProfileName
are the profile name used for the deidentificationProfileCodenames
are a list of concatenated profile items that has been used for the deidentificationYou must set the environment variable LOGBACK_CONFIGURATION_FILE
with the path of Logback file configuration, it will override the default log file.
For example, if you create your own configuration with the file name my-logback.xml
at the root of the docker-compose.yml.
You must create a volume that will copy the file in the Karnak docker and define the environment variable LOGBACK_CONFIGURATION_FILE
with the path of the file in the docker container.
services:
karnak:
container_name: karnak
image: osirixfoundation/karnak:v0.9.7
volumes:
- ./my-logback.yml:/logs/my-logback.xml
environment:
LOGBACK_CONFIGURATION_FILE: /logs/my-logback.xml
If you use directly the Karnak jar, you must add the following parameter at the startup:
-Dlogging.config=my-logback.xml
The default logback file (see below) has two modes.
ENVIRONMENT
must be set to DEV
to be activate.INFO
level except for the package org.karnak
and org.weasis
, they are at the DEBUG
level.ENVIRONMENT
is set to DEV
.WARN
levelWARN
level logorg.weasis
INFO
level logorg.karnak
INFO
level log, excepted clinical logCLINICAL
which concerns the deidentification<?xml version="1.0" encoding="UTF-8"?>
<configuration>
<property name="LOGS" value="logs" />
<!--
DEV LOGS
Condition to verify if "ENVIRONMENT" is "DEV"
-->
<if condition='property("ENVIRONMENT").contains("DEV")'>
<then>
<!--
How the logs will be displayed in the standard output (console)
SOPInstanceUID, issuerOfPatientID and PatientID are variables to associate the current DICOM with the log
-->
<appender name="DEV_OUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<Pattern>%black(%d{ISO8601}) %highlight(%-5level) %marker %highlight(%X{SOPInstanceUID}) %highlight(%X{issuerOfPatientID}) %highlight(%X{PatientID}) [%yellow(%t)] %yellow(%C{1.}): %msg%n%throwable</Pattern>
</encoder>
</appender>
<!--
Log everything at the INFO level except for the package org.karnak and org.weasis, they are at the DEBUG level.
-->
<root level="info">
<appender-ref ref="DEV_OUT" />
</root>
<logger name="org.weasis" level="debug" />
<logger name="org.karnak" level="debug" />
</then>
<!--
PRODUCTION LOGS
-->
<else>
<!--
How the warning logs will be displayed in the standard output (console)
SOPInstanceUID, issuerOfPatientID and PatientID are variables to associate the current DICOM with the log
-->
<appender name="WARNING_OUT" class="ch.qos.logback.core.ConsoleAppender">
<!--
Log only at WARN level
-->
<filter class="ch.qos.logback.classic.filter.ThresholdFilter">
<level>WARN</level>
</filter>
<encoder>
<Pattern>%black(%d{ISO8601}) %highlight(%-5level) %marker %highlight(%X{SOPInstanceUID}) %highlight(%X{issuerOfPatientID}) %highlight(%X{PatientID}) [%yellow(%t)] %yellow(%C{1.}): %msg%n%throwable </Pattern>
</encoder>
</appender>
<!--
Write all.log file in the file system.
-->
<appender name="ALL_LOGS" class="ch.qos.logback.core.rolling.RollingFileAppender">
<file>${LOGS}/all/all.log</file>
<!--
Window rolling policy defined by the variable KARNAK_LOGS_MIN_INDEX or KARNAK_LOGS_MAX_INDEX
By default the min is 1 and the max is 10
-->
<rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy">
<fileNamePattern>${LOGS}/all/all_%i.log</fileNamePattern>
<minIndex>${KARNAK_LOGS_MIN_INDEX:-1}</minIndex>
<maxIndex>${KARNAK_LOGS_MAX_INDEX:-10}</maxIndex>
</rollingPolicy>
<!--
Size rolling policy defined by the variable KARNAK_LOGS_MAX_FILE_SIZE
By default the max file size is 50MB
-->
<triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy">
<maxFileSize>${KARNAK_LOGS_MAX_FILE_SIZE:-50MB}</maxFileSize>
</triggeringPolicy>
<!--
Filter not to write logs with the clinical marker
-->
<filter class="ch.qos.logback.core.filter.EvaluatorFilter">
<evaluator class="ch.qos.logback.classic.boolex.OnMarkerEvaluator">
<marker>CLINICAL</marker>
</evaluator>
<onMismatch>NEUTRAL</onMismatch>
<onMatch>DENY</onMatch>
</filter>
<encoder>
<pattern>%d %-5level %m%n</pattern>
</encoder>
</appender>
<!--
Write clinical.log file in the file system.
Will contains only the logs with the CLINICAL marker. The logs with CLINICAL marker concerns the information about the deidentification.
Variables used for the CLINICAL log:
SOPInstanceUID, DeidentifySOPInstanceUID, SeriesInstanceUID, DeidentifySeriesInstanceUID, ProjectName, ProfileName, ProfileCodenames
-->
<appender name="CLINICAL_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
<file>${LOGS}/Clinical/clinical.log</file>
<rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy">
<fileNamePattern>${LOGS}/Clinical/clinical_%i.log</fileNamePattern>
<minIndex>${KARNAK_CLINICAL_LOGS_MIN_INDEX:-1}</minIndex>
<maxIndex>${KARNAK_CLINICAL_LOGS_MAX_INDEX:-10}</maxIndex>
</rollingPolicy>
<triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy">
<maxFileSize>${KARNAK_CLINICAL_LOGS_MAX_FILE_SIZE:-50MB}</maxFileSize>
</triggeringPolicy>
<filter class="ch.qos.logback.core.filter.EvaluatorFilter">
<evaluator class="ch.qos.logback.classic.boolex.OnMarkerEvaluator">
<marker>CLINICAL</marker>
</evaluator>
<onMismatch>DENY</onMismatch>
<onMatch>NEUTRAL</onMatch>
</filter>
<encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
<Pattern>%d SOPInstanceUID_OLD=%X{SOPInstanceUID} SOPInstanceUID_NEW=%X{DeidentifySOPInstanceUID} SeriesInstanceUID_OLD=%X{SeriesInstanceUID} SeriesInstanceUID_NEW=%X{DeidentifySeriesInstanceUID} ProjectName=%X{ProjectName} ProfileName=%X{ProfileName} ProfileCodenames=%X{ProfileCodenames}</Pattern>
</encoder>
</appender>
<!--
Log everything at the WARN level except for the package org.karnak and org.weasis, they are at the INFO level.
The INFO logs won't appear in the standard output, because they will be filtered by the WARNING_OUT appender
-->
<root level="warn">
<appender-ref ref="ALL_LOGS" />
<appender-ref ref="CLINICAL_FILE" />
<appender-ref ref="WARNING_OUT" />
</root>
<logger name="org.weasis" level="info" />
<logger name="org.karnak" level="info" />
</else>
</if>
</configuration>
Example of systemd service configuration with a docker-compose.yml file in the folder /opt/karnak (If it’s another directory you have to adapt the script).
By default, Docker needs root privileges to be executed.
Manage Docker as a non-root user
For more details, the following commands are inspired by the official Docker documentation.
Create the docker
group.
sudo groupadd docker
Add your user to the docker
group.
sudo usermod -aG docker $USER
Activate the changes to groups.
newgrp docker
Verify that you can run docker
commands without sudo
.
docker run hello-world
Specify User in the service
In the [Service]
section of the karnak.service (see below), it’s possible to specify the user who will run the service.
User=root
Instructions:
# /etc/systemd/system/karnak.service
#########################
# KARNAK #
# SERVICE #
##########################
[Unit]
Description=Docker Compose KARNAK Service
Requires=docker.service
After=docker.service network.target
[Service]
Type=oneshot
RemainAfterExit=yes
WorkingDirectory=/opt/karnak
ExecStart=/usr/local/bin/docker compose up -d
ExecStop=/usr/local/bin/docker compose down
TimeoutStartSec=0
[Install]
WantedBy=multi-user.target
Test the service:
Presentation of the Profile Structure
Definition of the actions applied to specific tags
Definition of the actions applied to dates
Definition of the expressions applied to specific tags
Definition of the masks applied to Pixel Data
Definition of the conditions applied to profile elements
Technical explanations about the de-identification process
In the Karnak user interface, the Profiles page can be accessed using the menu bar on the left. It displays the list of existing profiles and offers to import new profiles.
A profile file is one or a list of profile elements that are defined for a group of DICOM attributes and with a particular action. During de-identification or tag morphing, Karnak will apply the profile elements to the applicable DICOM attributes. Only one profile element can be applied to a DICOM attribute. The profile elements are applied in the order defined in the yaml file and, therefore, the first applicable profile element will modify the value of a DICOM attribute. If other profile elements were applicable to that specific tag, they won’t be applied since it has already been modified.
Currently, the profile must be a yaml file (MIME-TYPE: application/x-yaml) and respect the definition as below.
All these metadata are optional, but for a better user experience we recommend defining at least the name and the version. They will be used to identify and select your profile in a Project.
name
- The name of your profile
version
- The version of your profile
minimumKarnakVersion
- The version of Karnak when the profile has been imported
defaultIssuerOfPatientID
- Default value in case the IssuerOfPatientID value is not available in DICOM file, it is used to build the patient’s pseudonym when applying de-identification
profileElements
- The list of profile elements, the elements are applied accordingly to their position in the list
A profile element is defined as below in the yaml file.
name
- The name of your profile element
codename
- The codename represents the type of profile element. The available types of profiles elements and their codename are described in details in this documentation
condition
- A boolean condition that defines some requirements to apply this profile element
action
- The type of action that will be applied
option
- Required for certain types of profile elements, contains a single value
arguments
- Required for certain types of profile elements, contains a list of key-value pairs
tags
- List of tags or pattern that identifies the DICOM attributes this profile should be applied to
excludedTags
- List of tags or pattern that identifies the DICOM attributes this profile should not be applied to. These attributes can then be modified by another profile element if applicable
DICOM Tags can be defined in different formats: (0010,0010)
; 0010,0010
; 00100010
;
A tag pattern represent a group of tags and can be defined as follows: e.g. (0010,XXXX)
represent all the tags of group 0010.
The pattern (XXXX,XXXX)
targets all the DICOM attributes.
A condition can be added to any type of profile element. It contains an expression that will be evaluated for each tag the profile element is applied to.
The syntax and usage of these conditions is detailed in the Conditions page.
The content of the yaml file is validated upon import. If the structure or parameters are not defined correctly, detailed errors will be displayed to the user.
Please refer to the Profiles page for more information.
The Basic DICOM Profile is defined by DICOM to remove all the attributes that could contain Individually Identifying Information (III) about the patient or other individuals or organizations associated with the data. The details of this profile element can be found in the DICOM Standard.
Further details on this profile element and its implementation in Karnak can be found in the How does de-identification work? page.
We strongly recommend including this profile as basis for de-identification.
This profile element can be included in the profile definition by referencing its codename:
- name: "DICOM basic profile"
codename: "basic.dicom.profile"
Example of a complete and valid profile yaml file that applies only the Basic DICOM Profile:
name: "Dicom Basic Profile"
version: "1.0"
minimumKarnakVersion: "0.9.2"
defaultIssuerOfPatientID:
profileElements:
- name: "DICOM basic profile"
codename: "basic.dicom.profile"
This profile element applies an action on a tag or a group of tags defined by the user. Its codename is action.on.specific.tags
.
This profile element requires the following parameters:
name
: description of the action appliedcodename
: action.on.specific.tags
action
: Remove (X
) / Keep (K
)tags
: list of tags the action should be applied toThis profile element can have these optional parameters:
condition
: optional, defines a condition to evaluate if this profile element should be applied to this DICOM instanceexcludedTags
: list of tags that will be ignored by this actionIn this example, all the tags starting with 0028 will be removed excepted (0028,1199) which will be kept.
- name: "Remove tags"
codename: "action.on.specific.tags"
action: "X"
tags:
- "(0028,xxxx)"
excludedTags:
- "(0028,1199)"
- name: "Keep tags 0028,1199"
codename: "action.on.specific.tags"
action: "K"
tags:
- "0028,1199"
This profile element applies an action on a private tag or a group of private tags defined by the user. This action won’t be applied in case the tag is not private.
Its codename is action.on.privatetags
.
This profile element requires the following parameters:
name
: description of the action appliedcodename
: action.on.privatetags
action
: Remove (X
) / Keep (K
)This profile can have these optional parameters:
condition
: optional, defines a condition to evaluate if this profile element should be applied to this DICOM instancetags
: list of tags the action should be applied to. If not specified, the action is applied to all private tagsexcludedTags
: list of tags that will be ignored by this actionIn this example, all tags starting with 0009 will be kept and all the other private tags will be removed.
- name: "Keep private tags starting with 0009"
codename: "action.on.privatetags"
action: "K"
tags:
- "(0009,xxxx)"
- name: "Remove all private tags"
codename: "action.on.privatetags"
action: "X"
This profile element adds a tag if it is not already present in the instance. This action will be ignored if the tag is already present in the instance.
Its codename is action.add.tag
.
This profile element requires the following parameters:
name
: description of the action appliedcodename
: action.add.tag
arguments
:
value
: required value to set the tag’s value tovr
: VR of the tag, if not specified, its VR will be retrieved from DICOM Standardtags
: must contain exactly one tag, the one to addThis profile can have these optional parameters:
condition
: optional, defines a condition to evaluate if this profile element should be applied to this DICOM instanceRegarding the application of profile elements, if the tag is not initially present in the instance, then it will be added by this action. Further actions that match this tag won’t be applied. If the tag was already present in the instance, then the add action is ignored and a further action can be applied to that tag.
This feature is especially useful when applying masks to non-compliant SOPs by using the attribute Burned In Annotation. Please refer to the Cleaning Data Pixel Exceptions page for an exhaustive example.
In this example, we add the optional tag Recognizable Visual Features (0028,0302).
- name: "Add Recognizable Visual Features tag"
codename: "action.add.tag"
arguments:
value: "YES"
vr: "CS"
tags:
- "(0028,0302)"
Example of a complete and valid profile yaml file that applies the following profile elements in order:
name: "De-identification profile"
version: "1.0"
minimumKarnakVersion: "0.9.2"
defaultIssuerOfPatientID:
profileElements:
- name: "Remove tags"
codename: "action.on.specific.tags"
action: "X"
tags:
- "(0008,00XX)"
- "0010,00XX"
excludedTags:
- "0008,0008"
- "0008,0013"
- name: "Keep tags"
codename: "action.on.specific.tags"
action: "K"
tags:
- "0008,0008"
- "0008,0013"
- name: "Remove all private tags"
codename: "action.on.privatetags"
action: "X"
- name: "DICOM basic profile"
codename: "basic.dicom.profile"
This example keeps a Study Description tag if that as contain ‘R2D2’ value, keep the Philips PET private group and apply the basic DICOM profile.
The tag patterns (0073,xx00) and (7053,xx09) are defined in Philips PET Private Group by DICOM.
name: "Profile Example"
version: "1.0"
minimumKarnakVersion: "0.9.7"
profileElements:
- name: "Keep StudyDescription tag according to a condition"
codename: "action.on.specific.tags"
condition: "tagValueContains(#Tag.StudyDescription, 'R2D2')"
action: "K"
tags:
- "(0008,1030)"
- name: "Keep Philips PET Private Group"
codename: "action.on.privatetags"
action: "K"
tags:
- "(7053,xx00)"
- "(7053,xx09)"
- name: "DICOM basic profile"
codename: "basic.dicom.profile"
This profile element applies a specific action on a tag or a group of tags containing a date. This action can only be applied to the following Value Representations: Age String (AS), Date (DA), Date Time (DT) and Time (TM).
This profile element requires the following parameters:
name
: description of the action appliedcodename
: action.on.dates
option
: the action to be applied, detailed belowarguments
: additional parameters depending on the chosen optionThis profile can have these optional parameters:
condition
: optional, defines a condition to evaluate if this profile element should be applied to this DICOM instancetags
: list of tags the action should be applied to. If not specified, the profile element will be applied to all the tags that have a AS, DA, DT or TM VR in the instanceexcludedTags
: list of tags that will be ignored by this actionThe parameter option can have one of the following values:
shift
shift_range
shift_by_tag
date_format
The shift option applies a shift to a date according to the following required arguments:
seconds
: integer representing the number of seconds the shift operation should applydays
: integer representing the number of days the shift operation should applyIn the case of a shift action applied to an Age String (AS) Value Representation, the seconds and days will be added to the existing value.
In case of a shift action applied to a Date (DA), Date Time (DT) or Time (TM) Value Representation, the seconds and days will be subtracted to the existing value.
In this example, all the tags starting with 0010 and that have a AS, DA, DT or TM VR will be shifted by 30 seconds and 10 days.
- name: "Shift Date"
codename: "action.on.dates"
arguments:
seconds: 30
days: 10
option: "shift"
tags:
- "0010,XXXX"
The shift range option applies a random shift to a date parametrized by ranges defined by the user as follows:
max_seconds
(required): integer representing the upper bound of the number of seconds the shift operation should applymax_days
(required): integer representing the upper bound of the number of days the shift operation should applymin_seconds
(optional): integer representing the lower bound of the number of seconds the shift operation should apply, the default value is 0 is not specifiedmin_days
(optional): integer representing the lower bound of the number of days the shift operation should apply, the default value is 0 is not specifiedThe random operation is deterministic and reproducible based on the patient and the project. For more details about the usage of randomization in Karnak, please refer to the Karnak Project.
In this example, all the tags starting with 0008,002 and that have a AS, DA, DT or TM VR will be shifted randomly in a range of 0 to 60 seconds and 50 to 100 days.
- name: "Shift Range Date"
codename: "action.on.dates"
arguments:
max_seconds: 60
min_days: 50
max_days: 100
option: "shift_range"
tags:
- "0008,002X"
The date format option applies a partial deletion to a date depending on the specified option:
day
: this option will delete the day information and replace it with the first day of the monthmonth_day
: this option will delete the day and month information and replace it with the first month and first day of the monthThis action can only be applied to the following Value Representations: Date (DA) and Date Time (DT).
In this example, all the tags starting with 0008,003 and that have a DA or DT VR will have the day information removed.
- name: "Date format"
codename: "action.on.dates"
arguments:
remove: "day"
option: "date_format"
tags:
- "0008,003X"
For example, if the value contained in the tag is 20230512
, the output value will be 20230501
.
In this example, all the tags starting with 0008,003 and that have a DA or DT VR will have the day and month information removed.
- name: "Date format"
codename: "action.on.dates"
arguments:
remove: "month_day"
option: "date_format"
tags:
- "0008,003X"
For example, if the value contained in the tag is 20230512
, the output value will be 20230101
.
The shift by tag option applies a shift to a date according to a value contained in a specified DICOM tag. The arguments are detailed below:
seconds_tag
: tag that contains the number of seconds the shift operation should applydays_tag
: tag that contains the number of days the shift operation should applyBoth arguments are not required, but at least one of them must be specified in order to apply the action.
In this example, all the tags starting with 0010 and that have a AS, DA, DT or TM VR will be shifted by the number of days stored in the private tag (0015,0011)
.
- name: "Shift Date By Tag"
codename: "action.on.dates"
arguments:
days_tag: "(0015,0011)"
option: "shift_by_tag"
tags:
- "0010,XXXX"
Example of a complete and valid profile yaml file that applies the following profile elements in order:
name: "De-identification profile"
version: "1.0"
minimumKarnakVersion: "0.9.2"
defaultIssuerOfPatientID:
profileElements:
- name: "Shift Range Date with arguments"
codename: "action.on.dates"
arguments:
max_seconds: 60
min_days: 10
max_days: 50
option: "shift_range"
tags:
- "0008,0012"
- "0008,003X"
excludedTags:
- "0008,0030"
- "0008,0032"
- name: "Date Format"
codename: "action.on.dates"
arguments:
remove: "month_day"
option: "format_date"
tags:
- "0008,0023"
- "0008,0021"
- name: "Shift Date with arguments"
codename: "action.on.dates"
arguments:
seconds: 30
days: 10
option: "shift"
tags:
- "0010,XXXX"
excludedTags:
- "0010,1010"
- name: "DICOM basic profile"
codename: "basic.dicom.profile"
This profile element applies an expression to a tag or a group of tags defined by the user. An expression is based on Spring Expression Language (SpEL) and returns a value or an action according to a certain condition.
Its codename is expression.on.tags
.
This profile element requires the following parameters:
name
: description of the action appliedcodename
: expression.on.tags
arguments
: definition of the expressiontags
: list of tags the action should be applied toThis profile can have these optional parameters:
condition
: optional, defines a condition to evaluate if this profile element should be applied to this DICOM instanceexcludedTags
: list of tags that will be ignored by this actionThe expression is contained in the expr
argument. The expression will be executed and can either return an action that will be executed, or a null value that will do nothing and move on the next profile element.
Some custom variables are defined in the context of the expression, they contain information relative to the current attribute the profile element is being applied to:
tag
contains the current attribute tag, for example (0010,0010)vr
contains the current attribute VR, for example PNstringValue
contains the current element value, for example ‘John^Doe’ //TOCHECKSome constants are also defined to improve readability and reduce errors.
#Tag
is a constant that can be used to retrieve any tag integer value in the DICOM standard. For example, #Tag.PatientBirthDate
corresponds to (0010,0030).#VR
is a constant that can be used to retrieve any VR value in the DICOM standard. For example, #VR.LO
corresponds to the Long String type.Some utility functions are defined to work with the tags:
getString(int tag)
returns the value of the given tag in the DICOM instance, returns null if the tag is not present
tagIsPresent(int tag)
returns true if the tag is present in the DICOM instance, false otherwise
The actions are defined as functions, they are used to set the action and its parameters for the profile element.
The possible actions are:
ReplaceNull()
: Sets to empty the current tag valueReplace(String dummyValue)
: Replaces the current tag valueRemove()
: Removes the tagKeep()
: Keeps the tag unchangedUID()
: Replaces the current tag value with a newly generated UID and sets the tag’s VR to UIAdd(int tagToAdd, int vr, String value)
: Adds a tag to the DICOM instanceComputePatientAge()
: Replaces the current tag value with a computed value of the patient’s age at the time of the exam//TOCHECK calls api + add ?
If the current tag corresponds to the Patient Name attribute and its value is ‘John’, its content will be replaced by the value of the Institution Name attribute, otherwise the tag is kept unchanged. This example is applied to all the attributes of the DICOM instance, and returns either a Keep or Replace action for every tag, implying that the following profile elements, if any, will be ignored since all the attributes already had an action applied to them.
- name: "Expression"
codename: "expression.on.tags"
arguments:
expr: "stringValue == 'John' and tag == #Tag.PatientName? Replace(getString(#Tag.InstitutionName)) : Keep()"
tags:
- "(xxxx,xxxx)"
If the current tag value is UNDEFINED, it will be kept as is (and it will implicitly block any further potential modifications applied by other profile elements), otherwise the tag is removed. This expression is applied to 2 tags, (0010,0010) and (0010,0212).
- name: "Expression"
codename: "expression.on.tags"
arguments:
expr: "stringValue == 'UNDEFINED'? Keep() : Remove()"
tags:
- "(0010,0010)" #PatientName
- "(0010,0212)" #StrainDescription
Replacement of the Study Description tag value with a concatenation of the Institution Name and the Station Name of the DICOM instance.
- name: "Expression"
codename: "expression.on.tags"
arguments:
expr: "Replace(getString(#Tag.InstitutionName) + '-' + getString(#Tag.StationName))"
tags:
- "(0008,1030)" #StudyDescription
Computation of the patient’s age at the time of the exam.
- name: "Expression"
codename: "expression.on.tags"
arguments:
expr: "ComputePatientAge()"
tags:
- "(0010,1010)" #PatientAge
This profile applies defacing to the image data of the DICOM instance.
This profile can only be applied to images in the Axial orientation of the following SOP:
This profile element requires the following parameters:
name
: description of the action appliedcodename
: clean.recognizable.visual.features
condition
: optional, defines a condition to evaluate if this profile element should be applied to this DICOM instanceThis profile applies a mask defined by the user on the DICOM instance pixel data to remove identifying information burned in the image.
The details on the masks definition can be found below.
This profile is applied only on the following SOP:
Or if the tag value Burned In Annotation (0028,0301) is “YES”
This profile element requires the following parameters:
name
: description of the action appliedcodename
: clean.pixel.data
condition
: optional, defines a condition to evaluate if this profile element should be applied to this DICOM instanceThe condition
parameter can be used to exclude the images coming from a specific machine for example.
profileElements:
- name: "Clean pixel data"
codename: "clean.pixel.data"
condition: "!tagValueContains(#Tag.StationName,'ICT256')"
The mask definition requires the following parameters:
stationName
: source station name that is matched against the attribute Station Name in the DICOM instance. It allows the mask to be specific depending on the station that generated the image. The value can also be set to *
to match any station.color
: color of the mask in hexadecimalrectangles
: defines the list of rectangles to apply to mask identifying informationThe mask definition can have these optional parameters:
imageWidth
: mask specific to an image of a given width in pixels, this value will be matched against the value of the Columns attribute in the DICOM instanceimageHeight
: mask specific to an image of a given height in pixels, this value will be matched against the value of the Rows attribute in the DICOM instance.The selection of the mask based on the image size requires both attributes to be set, height and width. The definition of the width or height solely is not supported.
A rectangle is defined by the following required parameters:
x
: x coordinate of the upper left corner of the rectangley
: y coordinate of the upper left corner of the rectanglewidth
: width of the rectangleheight
: height of the rectangleThe upper left corner of the image corresponds to the coordinates (0,0).
The schema below illustrate the definition of a rectangle having the following parameters (25, 75, 150, 50).
The example below shows how to define a default mask (stationName: *
), a mask specific to the R2D2 station and a more specific mask applied only depending on the image size.
Depending on the instance image size and station name, the following actions will be performed:
masks:
- stationName: "*"
color: "ffff00"
rectangles:
- "25 75 150 50"
- stationName: "R2D2"
color: "00ff00"
rectangles:
- "25 25 150 50"
- "350 15 150 50"
- stationName: "R2D2"
imageWidth: 1024
imageHeight: 1024
color: "00ffff"
rectangles:
- "50 25 100 100"
In some cases, often based on the manufacturer and equipment, pixel data can contain embedded identifying information. Below is an exhaustive example illustrating how to apply cleaning pixel profile element depending on the station that produced the image, applicable to any DICOM modality.
name: "Clean pixel data"
version: "1.0"
minimumKarnakVersion: "0.9.2"
defaultIssuerOfPatientID:
profileElements:
- name: "Add tag BurnedInAnnotation if does not exist"
codename: "action.add.tag"
condition: "tagValueContains(#Tag.StationName, 'ICT256') && !tagIsPresent(#Tag.BurnedInAnnotation)"
arguments:
value: "YES"
vr: "CS"
tags:
- "(0028,0301)"
- name: "Set BurnedInAnnotation to YES"
codename: "expression.on.tags"
condition: "tagValueContains(#Tag.StationName, 'ICT256')"
arguments:
expr: "Replace('YES')"
tags:
- "(0028,0301)"
- name: "Clean pixel data"
codename: "clean.pixel.data"
- name: "DICOM basic profile"
codename: "basic.dicom.profile"
masks:
- stationName: "*"
color: "ffff00"
rectangles:
- "25 75 150 50"
- stationName: "ICT256"
color: "00ff00"
rectangles:
- "25 25 150 50"
- "350 15 150 50"
A condition is an expression evaluated in a certain context and that returns a boolean value (true or false).
Some constants are also defined to improve readability and reduce errors.
#Tag
is a constant that can be used to retrieve any tag integer value in the DICOM standard. For example, #Tag.PatientBirthDate
corresponds to (0010,0030).#VR
is a constant that can be used to retrieve any VR value in the DICOM standard. For example, #VR.LO
corresponds to the Long String type.Utility functions are available to define the conditions and are detailed below.
tagValueIsPresent(int tag, String value)
or tagValueIsPresent(String tag, String value)
This function will retrieve the tag
value of the DICOM and check if the value
parameter is the same as the tag value.
# Check if the study description is equals to "755523-st222-GE"
tagValueIsPresent(#Tag.StudyDescription, "755523-st222-GE")
tagValueIsPresent("0008,1030", "755523-st222-GE")
# Check if the study description is not equals to "755523-st222-GE"
!tagValueIsPresent(#Tag.StudyDescription, "755523-st222-GE")
!tagValueIsPresent("0008,1030", "755523-st222-GE")
tagValueContains(int tag, String value)
or tagValueContains(String tag, String value)
This function will retrieve the tag
value of the DICOM and check if the value
parameter appears in the tag value.
# Check if the study description contains "st222"
tagValueContains(#Tag.StudyDescription, "st222")
tagValueContains("0008,1030", "st222")
# Check if the study description does not contain "st222"
!tagValueContains(#Tag.StudyDescription, "st222")
!tagValueContains("0008,1030", "st222")
tagValueBeginsWith(int tag, String value)
or tagValueBeginsWith(String tag, String value)
This function will retrieve the tag
value of the DICOM and check if the tag value begins with the parameter value
# Check if the study description begins with "755523"
tagValueBeginsWith(#Tag.StudyDescription, "755523")
tagValueBeginsWith("0008,1030", "755523")
# Check if the study description does not begin with "755523"
!tagValueBeginsWith(#Tag.StudyDescription, "755523")
!tagValueBeginsWith("0008,1030", "755523")
tagValueEndsWith(int tag, String value)
or tagValueEndsWith(String tag, String value)
This function will retrieve the tag
value of the DICOM and check if the tag value ends with the parameter value
# Check if the study description ends with "GE"
tagValueEndsWith(#Tag.StudyDescription, "GE")
tagValueEndsWith("0008,1030", "GE")
# Check if the study description does not end with "GE"
!tagValueEndsWith(#Tag.StudyDescription, "GE")
!tagValueEndsWith("0008,1030", "GE")
tagIsPresent(int tag)
or tagIsPresent(String tag)
This function will check if the tag
is present in the DICOM instance.
# Check if the tag study description is present in the DICOM file
tagIsPresent(#Tag.StudyDescription)
tagIsPresent("0008,1030")
# Check if the tag study description is not present in the DICOM file
!tagIsPresent(#Tag.StudyDescription)
!tagIsPresent("0008,1030")
Multiple conditions can be combined using logical operators.
&&
corresponds to the AND logical operator
||
corresponds to the OR logical operator
# Check if the tag study description ends with "GE" and if the station name is "CT1234"
tagValueEndsWith(#Tag.StudyDescription, "GE") && tagValueContains(#Tag.StationName, "CT1234")
# Check if the tag study description ends with "GE" or if the station name is "CT1234"
tagValueEndsWith(#Tag.StudyDescription, "GE") || tagValueContains(#Tag.StationName, "CT1234")
Karnak is a gateway for sending DICOM files to one or multiple Application Entity Title (AET). Karnak offers the possibility to configure multiple destinations for an AET. These destinations can communicate using the DICOM or DICOM WEB protocol.
A destination contains several configurations for the DICOM endpoint, the credentials, and is also linked to a project.
A project defines the de-idenfication or tag morphing method and a secret that will be used to generate deterministic random values like UIDs or shift date arguments.
The reference profile for de-identifying DICOM objects is provided by the DICOM standard. This profile defines an exhaustive list of DICOM tags and their related action to allow the de-identification of the instance.
In order to properly de-identify the sensitive data, five different actions are defined in the standard:
The Basic Profile defines one or more actions to be applied to a list of tags.
The DICOM’s type is often dependent on the Information Object Definition (IOD) of the instance. To avoid DICOM corruption, multiple actions can be defined for a tag, ensuring that a destructive action like REMOVE won’t be applied on a Type 1 or Type 2 attribute.
Karnak loads the SOPs and attributes as specified in the DICOM Standard. Based on the tag’s type in the current instance, the proper action is set and applied. If the tag cannot be identified in the SOP or its type cannot be inferred, the strictest action will be applied (U/D > Z > X).
Below is a concrete illustration of the action applied in case of multiple actions defined in the Basic Profile:
The action D replaces the tag value with a dummy one. This value must be consistent with the Value Representation (VR) of the tag.
Karnak will use a default value based on the VR in this case, as defined below:
The shiftRange() action will return a random value between a given maximum days and seconds. By default, maximum days is set to 365 and maximum seconds is set to 86400.
The following VRs FL, FD, SL, SS, UL, US are of type Binary. By default, Karnak will set to null the value of this VR.
For each U action, Karnak will hash the input value. A one-way function is created to ensure that it is not possible to revert to the original UID. This function will hash the input UID and generate a new UID from the hashed input.
It’s possible for a DICOM study to be de-identified several times and in different ways, potentially implying the use of multiple hashing and one-way functions. Karnak ensures deterministic generation of UIDs in order to maintain the quality and usability of the data.
To achieve that behavior, a project must be created and associated to the destination that requires de-identification. A project defines a de-idenfication method and a secret, either generated randomly or imported by the user. The project’s secret will be used as key for the HMAC.
The secret is 16 bytes long and randomly defined when the project is created.
A user can upload his own secret, but it must be 16 bytes long. It can be uploaded as a String in hexadecimal format.
The algorithm used for hashing is the “Message Authentication Code” (MAC). Karnak uses the MAC, not as message authentication, but as a one-way function. Below is a definition from the JAVA Mac class used in Karnak:
« A MAC provides a way to check the integrity of information transmitted over or stored in an unreliable medium, based on a secret key. Typically, message authentication codes are used between two parties that share a secret key in order to validate information transmitted between these parties.
A MAC mechanism that is based on cryptographic hash functions is referred to as HMAC. HMAC can be used with any cryptographic hash function, e.g., SHA256 or SHA384, in combination with a secret shared key. HMAC is specified in RFC 2104. »
For each use of the HMAC, it uses the SHA256 hash function combined with the project’s secret.
« What DICOM calls a “UID” is referred to in the ISO OSI world as an Object Identifier (OID) » [1]
To generate a new DICOM UID, Karnak will create an OID beginning with “2.25”, that is an OID encoded UUID.
« The value after “2.25.” is the straight decimal encoding of the UUID as an integer. It MUST be a direct decimal encoding of the single integer, all 128 bits. » [2]
The generated UUID will use the first 16 bytes (128 bits) from the hash value. The UUID is a type 4 with a variant 1. See the pseudocode below to ensure the type and the variant are correct in the UUID:
// Version
uuid[6] &= 0x0F
uuid[6] |= 0x40
// Variant
uuid[8] &= 0x3F
uuid[8] != 0x80
The hashed value will be converted in a positive decimal number and appended to the OID root separated by a dot. See the example below:
OID_ROOT = “2.25”
uuid = OID_ROOT + “.” + HashedValue[0:16].toPositiveDecimal()
Karnak implements a randomized date shifting action. This shift must be identical based on the project and the patient for data consistency. For example, if a random shift is made for the birthdate of the patient “José Santos”, it must be the same for each instance associated to “José Santos”, even if the instance is loaded later.
The random shift date action will use the HMAC defined above and a range of days or seconds defined by the user. If the minimum isn’t specified, it defaults to 0.
The patientID, along with the project’s secret, will be passed to the HMAC, ensuring data consistency by patient.
The code below illustrates how a random value is generated within a given minimum (inclusive) and maximum (exclusive) range.
scaleHash(PatientID, scaledMin, scaledMax):
patientHashed = hmac.hash(PatientID)
scale = scaledMax - scaledMin
shift = (patientHashed[0:6].toPositiveDecimal() * scale) + scaledMin
This chapter details the problems linked to the PatientID generation for different de-identification methods.
A patient can participate in several studies using different de-identification methods. Depending on the project or the clinical research, the de-identification profile can keep some of the patient’s metadata. A pseudonym and a patientID are generated and affected to the patient in order to identify him in the context of the project.
Most of the patient’s identifying information is contained in the Patient Module.
Below is illustrated a case where some patient’s data could be leaked. During the de-identification, the patient is associated with a pseudonym provided by an external service or mapping table. In this example, a patient’s study falls in the scope of two different projects in Karnak. The first de-identification removes the patient birthdate (3. Apply Project 1) and the second keeps it (5. Apply Project 2). If the patient pseudonym is used as patient identification in the Patient Module, and the data is reconciled between the two study, the birthdate will be leaked.
Below is illustrated an alternative way of handling pseudonyms during the de-identification.
In this example, the patient is still associated with a pseudonym provided by an external service or mapping table. But Karnak will generate a PatientID based on this pseudonym and other characteristics linked to the project specifically. This ID will be used to identify the patient in the context of the project. In case of different de-identification methods in different projects, the patients cannot be reconciled and data won’t be leaked.
Karnak generates a PatientID to solve the problem explained previously. The PatientID is generated using the HMAC function defined in the project, see chapter Action U, Generate a new UID for more details.
The de-identified patientID is generated as follows :
The pseudonym will be used as the patient’s name if no other action has been defined during de-identification.
It is possible to retrieve the patient information once de-identified.
The patientID is generated from the pseudonym and the project’s secret using the hmac function, making it irreversible. The pseudonym is stored in the attribute Clinical Trial Subject ID (0012,0040). Using the mapping table or the service that provided the pseudonym, it is possible to retrieve the original patient identity.
« The Clinical Trial Subject ID (0012,0040) identifies the subject within the investigational protocol specified by Clinical Trial Protocol ID (0012,0020). » [3]
Some attributes are automatically set by Karnak during de-identification in the following modules.
The following attributes are set in the SOP Common module during de-identification:
The following attributes are set in the SOP Common module during de-identification:
The profile element codenames are concatenated and separated by -. For example, a profile composed of the profile elements action.on.specific.tags and basic.dicom.profile will appear as action.on.specific.tags-basic.dicom.profile.
The following attributes are set in the Clinical Trial Subject Module during de-identification:
This user guide provides a detailed overview of each page available in the Karnak web interface, focusing on the sections listed in the left-hand menu.
Gateway management in Karnak
Manage the profiles
Manage the projects
Imports pseudonyms from a CSV file or manually
Query the cached pseudonyms to retrieve the original data
Log of the completed or ongoing transfers
DICOM tools for checking the connection to a DICOM server
Send DICOM data to a Kheops album
This page allows the creation of a forward node and its edition. All the forward nodes present in your Karnak are listed here. The forward node will configure the Application Entity of your DICOM node.
1. Creation of a forward node
Once clicked on the button “New forward node”, a form will appear, as below, instead of the “New forward node” button (It will replace it).
To create a new forward node, you have to complete the “Forward AETitle” input and click on the “Add” button. Once added, it will appear in the list of forward nodes and will be selected. A Forward AETitle must be unique.
2. Forward node list
This list displays all the forward nodes created in your Karnak instance. You can select a forward node to view and manage its details on the right panel.
3. Forward node inputs
You can directly modify the value of the forward AETitle and its description. To save your changes, you have to click on the “Save” action button.
4. Forward node sources or destinations
In an Application Entity, you can add source control, which checks if the received DICOM is provided from a known source. You can also create one or more destinations to distribute received DICOM instance. These destinations can communicate with the DICOM or DICOM WEB protocol.
4.1 Sources or destinations navigation
Depending on the navigation selected, between sources and destinations, the list and the action buttons will change.
4.2 Filter in destinations or sources list
The destination filter will use the destination description.
The source filter will use the AE title, hostname and description source.
4.3 Destination or sources list
All the destinations or sources, depending on the navigation selected, associated to the forward node is listed here.
You can click on a destination or source present in this list to view details or to modify it.
4.4 Action buttons
The action buttons will change depending the navigation selected.
For a destination, these buttons defined the protocol you can use to create a destination. The protocol can be DICOM or DICOM WEB. The details for the destination creation will be explained below in the section Destinations.
If you click on the DICOM button, the DICOM destination creation page will appear.
If you click on the STOW button, the DICOM WEB destination creation page will appear.
For a source, the illustration below show the button. The details for the source creation will be explained below in the section Sources.
If you click on the the Sources button, the source creation page will appear.
5. Action buttons
You have three action buttons available.
Beware, to activate the de-identification in a destination, you must first create a new project.
You can create a destination that sends your studies using the DICOM or STOW protocol.
For a DICOM destination the following is mandatory:
1. Destination field
These fields define the destination to which Karnak should send.
The Condition is a field to define a condition for the destination. If the condition is met, the destination will be activated. See Destination Conditions for more details.
The hostname and the port will be used to define the host in case the “Use AETitle destination” is not checked.
2. Transfer Syntax
This field will define the transfer syntax used.
3. Use AETitle destination
If “Use AETitle destination” is checked, the AETitle defined will be used as host.
4. Notifications
These fields will allow you to define the emails to be notified during various events that take place during the sending.
5. De-identification
The de-identification in the DICOM destination will be explained below, in the section Activate the de-identification
6. Authorized SOPs
This field allows you to define a filter on a list of SOPs. The chosen SOPs will serve as a filter for Karnak when sending. This allow you to define a destination which is applied only for SOPs of your choice. The SOPs present in the list are defined by DICOM.
If “Authorized SOPs” is checked, you can define one or more SOPs.
If “Authorized SOPs” is not checked, Karnak will not apply a SOP filter and will let all SOPs pass.
The following illustration shows the list of SOPs with two SOPs selected.
7. Enable the destination
This field allows you to enable or disable a destination.
If “Enable destination” is checked, the destination is enable.
If “Enable destination” is not checked, the destination is disable.
8. Actions buttons
You have three action buttons available.
For a STOW destination the following is mandatory:
1. Destination field
The URL is the DICOM endpoint of your final destination.
The Condition is a field to define a condition for the destination. If the condition is met, the destination will be activated. See Destination Conditions for more details.
The URL credentials is the STOW-RS service (format is "user:password")
2. Destination headers
This field will contains the headers for HTTP request.
To add the header Authorization: Bearer 123456790 you must write following the format below:
<key>Authorization</key>
<value>Bearer 1234567890</value>
3. Transfer Syntax
This field will define the transfer syntax used.
4. Notifications
These fields will allow you to define the emails to be notified during various events that take place during the sending.
5. De-identification
The de-identification in the STOW destination will be explained below, in the section Activate the de-identification
6. Authorized SOPs
This field allows you to define a filter on a list of SOPs. The chosen SOPs will serve as a filter for Karnak when sending. This allow you to define a destination which is applied only for SOPs of your choice. The SOPs present in the list are defined by DICOM.
If “Authorized SOPs” is checked, you can define one or more SOPs.
If “Authorized SOPs” is not checked, Karnak will not apply a SOP filter and will let all SOPs pass.
The following illustration shows the list of SOPs with two SOPs selected.
7. Switching in different Kheops albums
You have the possibility of sharing your data in different Kheops album. For this you must use a DICOM endpoint from Kheops.
The purpose of this functionality is to allow sending your data to a single destination and to use the Kheops API to propagate your data to different places without having to create a new destination.
The following illustration show a scenario of this functionality. The illustrated scenario allows you to send a DICOM data to Karnak. Karnak has a destination defined to send the data to a Kheops album (Album main). This means that this album will regroup all the data sent by KANRAK. To prevent researchers or end users from having access to all the data, the data will be shared in other albums according to defined conditions.
Details for configuring this feature are explained in the Kheops chapter.
8. Enable the destination
This field allows you to enable or disable a destination.
If “Enable destination” is checked, the destination is enable.
If “Enable destination” is not checked, the destination is disable.
9. Actions buttons
You have three action buttons available.
To use de-identification, a project must be present. See create a new project if you haven’t already created a project.
When you created a new destination, “Activate the de-identification” is not checked. When you check the box, Karnak will react in two different ways depending on whether you have a project or not.
If you have not yet created a project a pop up will appear asking you to create a project, as illustrated below.
Create a project will redirect you to the page project. Beware, if you click on this button your configuration already made will be lost, save it first.
Continue will close the pop-up and the de-identification will be unchecked.
If you have already create the project, you can configure the de-identifcation.
1. Project
Here you can select the project to use for de-identification.
A message below shows you which profile is associated with the selected project.
2. Pseudonym Type
2.1 By default, you can choose four different type of pseudonym.
2.2 Is an exemple of external id provider injected in Karnak (for more details, see ExternalID provider).
Pseudonym is generated by Mainzelliste will generate the pseudonym and store it by using Mainzelliste.
Pseudonym that has already been entered manually in Mainzelliste will use the pseudonym that an user has added by using the mainzelliste pseudonym page. Pseudonym are stored in mainzelliste.
Pseudonym is already store in Karnak will use the pseudonym that an user has added by using the pseudonym page. Pseudonym added alive in the Karnak cache. Beware if the patient to be de-identified is not present in the Karnak cache, it will not be sent.
Pseudonym is in a DICOM tag, will look for the pseudonym in a DICOM tag. The illustration below shows the different elements to configure.
1 Use as Patient Name
Instead to use the generated patient name by Karnak, it will contains the pseudonym find in the DICOM tag.
2 Fields to find the pseudonym
This fields are used to define the configuration to find the pseudonym.
Tag - will contain the tag where Karnak should look for the pseudonym.
Delimiter - will contain the delimiter to use for split the value of the tag. (Optionnal)
Position - will contain the position where the pseudonym is after the split. (Optionnal)
In case the delimiter and the position fields are not defined, Karnak will directly use the value present in the given tag.
3 Save the pseudonym
If “Save pseudonym” is checked, Karnak will save the patient and his pseudonym in Mainzelliste.
A source is used to perform control over the sending entity to the associated forward node. It will check if the received DICOM is provided from an AET known.
In the case that no source is defined, the verification does not take place and the forward node can receive data from any AET.
1. Form input
All the fields are optional, excepted the AETitle, which is mandatory.
If the “Check the hostname” box is checked, Karnak also checks the hostname.
2. Action buttons
You have three action buttons available.
The profile page allows you to manage your profiles. On this page.
Karnak has the “Dicom Basic Profile” profile by default, the details about this profile is explained in the section How does de-identification work?. This profile cannot be deleted or edited, except for the value of default issuer of patient ID.
As presented below, you can drag and drop your yaml profile configuration.
All the profiles available in your Karnak instance are listed here. To view Profile details, you can click on a profile.
This panel will change depending on your action.
Once you have clicked on a profile, this panel will display the details about the chosen profile.
If you have drag and drop your profile, the details about the profile uploaded will be presented. In case your profile contains any errors, it will show where and what the error are, as the illustration below.
The following table shows you the different action of icons present in the profile details.
When you want to use Karnak’s de-identification methods, you need to associate a project with your destination. This project defines the profile to be used as weel as the integrity of the data using a secret. The secret makes it possible to maintain consistency between the data to be de-identified, see Action U, Generate a new UID for more details.
The project page allows you to manage your projects. On this page, you can perform the following actions:
To create a new project, you must give a project name and choose a profile that will be used for de-identification. To validate your choose, you can click on the button “Add”.
The list of profiles that you can associated with a project are the profiles that you have uploaded, see Profile drag and drop area.
All the projects available in your Karnak instance are listed here. You can click on a project to view its details in the right panel.
You can change the project name and the de-identification profile using these inputs. By clicking on the “Update” button, see Action buttons, you save your changes.
The secret is generated automatically at the creation of a project. You can generate a new secret by using the “Generate Secret” button.
If new secrets are generated, previous secrets are saved in database in order to be able to use them again if needed.
The history of secrets is listed in the secret combobox with the creation date.
In order to take into account the new generated secret, project has to be saved by clicking on the update button.
A secret is a hexadecimal value
Must contains 32 characters
Button | Action |
---|---|
![]() ![]() |
Save your changes apply to project inputs |
![]() ![]() |
Remove the project. However, if a project is used by a destination, you will not be able to delete it |
Karnak will always use a given pseudonym when de-identifying. In this page, it is possible to import your data manually or from a CSV file your existing pseudonyms. So that Karnak can de-identify your studies while using your pseudonyms.
The pseudonyms are stored for a short period of time. Your data is stored in the cache for maximum 7 days. If Karnak is restarted, the uploaded data will be lost.
Each external pseudonym is linked to a project. In this way, there is no risk of collision between different external pseudonyms, if they are two similar pseudonym but patient information is different.
To upload a CSV file containing external pseudonyms, you must click in the “Upload File” button.
A pop-up will appear and offer to enter the separator character of the CSV file. In the exemple bellow, we use the “,” separator.
After click “Open CSV” button, a grid will be appear with the CSV file content.
This input field allows you to indicate from which line you want to start reading the CSV file. Your CSV file may contain headers. If these headers are in line 1 like the above example (EXTERNALID, PATIENTID, LASTNAME, FIRSTNAME, ISSUER_OF_PATIENTID), you can specify the entry “From line” to the value 2.
Fill in the correct fields with the corresponding columns. All the fields are mandatory, excepted the Issuer of patient ID.
Click to the “Upload CSV” button to validate the fields with the correct columns to store the externals pseudonyms in Karnak.
It’s also possible to add the external pseudonym manually. All the fields are mandatory, excepted the Issuer of patient ID.
As explained in the introduction, the pseudonyms are stored for a short period of time in Karnak. It’s possible to edit the patient fields by clicking “Edit"button or delete a patient by clicking “Delete” button.
The pseudonym mapping is used to retrieve the mapping of pseudonyms added in the views External pseudonym and Mainzelliste pseudonym.
In order to retrieve the mapping enter the desired pseudonym and click on the Find button.
If found, the external or mainzelliste pseudonym mapping will be displayed.
Otherwise some messages will indicate no mapping found.
Monitoring view is used to follow transfers in progress or recent transfers.
Currently limited to 150000 transfers, it is possible to browse the different pages via the navigation bar at the bottom center of the view
An automatic cleaning occurs on the monitoring tables, if the limit is exceeded.
The view display the most recent transfers and is ordered with most recent first.
Filters allow to limit the results and makes it easier to find a transfer.
It is possible to export the list of transfers via the export functionality in csv format depending on the filters selected.
By clicking on a transfer, it opens details of the information original and deidentified.
Filters allow to ease the search of transfers. It is possible to filter by:
It is possible to go throw the different pages of the list of transfers by using the navigation bar.
By clicking on the refresh button, the list of transfers is updated with last transfers.
Export settings allow to customize the csv before exporting it. It is possible to change the csv delimiter and the quote character of the file.
Once customize, the export is launched by clicking on the export button.
Export will use the filters selected in the monitoring view and export only transfers corresponding to the filters.
To create a Kheops album as a destination you must use:
First create the album destination, please refer to the official documentation of Kheops to create a new album.
1 Create a new token
2 Give WRITE permission to the token
3 Set a token expiration date
4 Copy the token value to be used in the header of your Karnak destination
Below is an example of creating headers with the token value:
<key>Authorization</key>
<value>Bearer y8KlxLhhq8yeEPpOHkhbHg</value>
When you create a destination that points to a Kheops album, you can propagate your data to underlying albums.
This is useful when you want to send a cohort of studies to a research group for example, without sharing all of the album studies.
Beware, the study sharing between album, must be done only within the same Kheops. Studies cannot be shared between different Kheops instances, you should create one destination per Kheops instance.
The purpose of this functionality is to allow sending your data to a single destination and to use the Kheops API to propagate your data to different places without having to create a new destination.
The following illustration show a scenario of this functionality. The illustrated scenario allows you to send a DICOM data to Karnak. Karnak has a destination defined to send the data to a Kheops album (Album main). This means that this album will regroup all the data sent by KANRAK. To prevent researchers or end users from having access to all the data, the data will be shared in other albums according to defined conditions.
To share your DICOM in different Kheops album, you must complete the following fields and validate them by clicking on Add button.
The destination is the album where the studies will be shared.
The source is the album main, where all studies are sent.
Fields | Description |
---|---|
Url API | The url of the Kheops API |
Valid token of destination | The token to write to the album destination. Need WRITE permission |
Valid token of source | The token to shared from the album source. Need READ, SEND (Sharing in the Kheops UI) permission |
The condition field will allow you to assign a condition to enable sharing to the destination.
The conditions syntax and usage is detailed in the Conditions page.