pydicom command-line interface
Command-line Interface Guide¶
Added in version 2.2.
Introduction¶
Starting in v2.2, pydicom offers a useful command-line interface (CLI) for exploring DICOM files, and access to the codify option for creating pydicom Python code. Additional subcommands may be added over time.
Example at the command line in a terminal window:
$ pydicom show pydicom::rtplan.dcm
Dataset.file_meta -------------------------------
(0002, 0000) File Meta Information Group Length UL: 156
(0002, 0001) File Meta Information Version OB: b'\x00\x01'
(0002, 0002) Media Storage SOP Class UID UI: RT Plan Storage
(0002, 0003) Media Storage SOP Instance UID UI: 1.2.999.999.99.9.9999.9999.20030903150023
(0002, 0010) Transfer Syntax UID UI: Implicit VR Little Endian
(0002, 0012) Implementation Class UID UI: 1.2.888.888.88.8.8.8
-------------------------------------------------
(0008, 0012) Instance Creation Date DA: '20030903'
(0008, 0013) Instance Creation Time TM: '150031'
(0008, 0016) SOP Class UID UI: RT Plan Storage
(0008, 0018) SOP Instance UID UI: 1.2.777.777.77.7.7777.7777.20030903150023
(0008, 0020) Study Date DA: '20030716'
...
Note that prefixing the file specification with pydicom::
will read the file
from the pydicom test data files rather than from the normal file system.
The following examples will use that so that you can replicate these
examples exactly. In normal use, you would leave the pydicom::
prefix
off when working with your files.
You can also show just parts of the DICOM file by specifying a data element using the usual pydicom keyword notation:
$ pydicom show pydicom::rtplan.dcm::FractionGroupSequence[0]
(300a, 0071) Fraction Group Number IS: "1"
(300a, 0078) Number of Fractions Planned IS: "30"
(300a, 0080) Number of Beams IS: "1"
(300a, 00a0) Number of Brachy Application Setups IS: "0"
(300c, 0004) Referenced Beam Sequence 1 item(s) ----
(300a, 0082) Beam Dose Specification Point DS: [239.531250000000, 239.531250000000, -751.87000000000]
(300a, 0084) Beam Dose DS: "1.0275401"
(300a, 0086) Beam Meterset DS: "116.0036697"
(300c, 0006) Referenced Beam Number IS: "1"
---------
You can see the available subcommands by simply typing pydicom
with no
arguments, or with pydicom help
:
$ pydicom help
Use pydicom help [subcommand] to show help for a subcommand
Available subcommands: codify, show
And, as noted in the block above, you get help for a particular subcommand
by typing pydicom help [subcommand]
. For example:
$ pydicom help show
usage: pydicom show [-h] [-x] [-t] [-q] filespec
Display all or part of a DICOM file
positional arguments:
filespec File specification, in format [pydicom::]filename[::element]. If `pydicom::` prefix is present,
then use the pydicom test file with that name. If `element` is given, use only that data element
within the file. Examples: path/to/your_file.dcm, your_file.dcm::StudyDate,
your_file.dcm::(0001,0001), pydicom::rtplan.dcm::BeamSequence[0],
yourplan.dcm::BeamSequence[0].BeamNumber, pydicom::rtplan.dcm::(300A,00B0)[0].(300A,00B6)
optional arguments:
-h, --help show this help message and exit
-x, --exclude-private
Don't show private data elements
-t, --top Only show top level
-q, --quiet Only show basic information
Installing the pydicom CLI¶
The pydicom
command should automatically be available after you
pip install pydicom. It should not require any updates to the system
path or environment variables.
If you are helping develop pydicom code, and are using git clones,
you will have to pip install -e .
from
the pydicom repository root. This has to be repeated for any changes to
pyproject.toml (e.g. to add a new subcommand).
If you are developing subcommands within your own package, you will need to reinstall your package similar to the above as you add entry points.
Combining with other CLIs¶
CLIs are useful for general exploration while programming, but also can be combined with other command-line filters for additional functionality. The following is an example of piping the output of the pydicom ‘show’ subcommand into ‘grep’, filtering for lines with either “Dose” or “Sequence” in them:
$ pydicom show pydicom::rtplan.dcm | grep "Dose\|Sequence"
(300a, 0010) Dose Reference Sequence 2 item(s) ----
(300a, 0012) Dose Reference Number IS: "1"
(300a, 0014) Dose Reference Structure Type CS: 'COORDINATES'
(300a, 0016) Dose Reference Description LO: 'iso'
(300a, 0018) Dose Reference Point Coordinates DS: [239.531250000000, 239.531250000000, -741.87000000000]
(300a, 0020) Dose Reference Type CS: 'ORGAN_AT_RISK'
(300a, 0023) Delivery Maximum Dose DS: "75.0"
(300a, 002c) Organ at Risk Maximum Dose DS: "75.0"
(300a, 0012) Dose Reference Number IS: "2"
(300a, 0014) Dose Reference Structure Type CS: 'COORDINATES'
(300a, 0016) Dose Reference Description LO: 'PTV'
(300a, 0018) Dose Reference Point Coordinates DS: [239.531250000000, 239.531250000000, -751.87000000000]
(300a, 0020) Dose Reference Type CS: 'TARGET'
(300a, 0026) Target Prescription Dose DS: "30.826203"
(300a, 0070) Fraction Group Sequence 1 item(s) ----
(300c, 0004) Referenced Beam Sequence 1 item(s) ----
(300a, 0082) Beam Dose Specification Point DS: [239.531250000000, 239.531250000000, -751.87000000000]
(300a, 0084) Beam Dose DS: "1.0275401"
(300a, 00b0) Beam Sequence 1 item(s) ----
(300a, 00b6) Beam Limiting Device Sequence 2 item(s) ----
(300a, 0111) Control Point Sequence 2 item(s) ----
(300a, 0115) Dose Rate Set DS: "650.0"
(300a, 011a) Beam Limiting Device Position Sequence 2 item(s) ----
(300c, 0050) Referenced Dose Reference Sequence 2 item(s) ----
(300a, 010c) Cumulative Dose Reference Coefficie DS: "0.0"
(300c, 0051) Referenced Dose Reference Number IS: "1"
(300a, 010c) Cumulative Dose Reference Coefficie DS: "0.0"
(300c, 0051) Referenced Dose Reference Number IS: "2"
(300c, 0050) Referenced Dose Reference Sequence 2 item(s) ----
(300a, 010c) Cumulative Dose Reference Coefficie DS: "0.9990268"
(300c, 0051) Referenced Dose Reference Number IS: "1"
(300a, 010c) Cumulative Dose Reference Coefficie DS: "1.0"
(300c, 0051) Referenced Dose Reference Number IS: "2"
(300a, 0180) Patient Setup Sequence 1 item(s) ----
(300c, 0002) Referenced RT Plan Sequence 1 item(s) ----
(300c, 0060) Referenced Structure Set Sequence 1 item(s) ----
Using the “or Sequence” (`\|Sequence`
) regular expression as above allows you
to see any filtered results in relation to their parent Sequences.
See the pydicom show command section for more examples of the show command, its options, and the ability to show only data elements or sequences within the file.
pydicom show
command¶
The pydicom show command displays representation of DICOM files or parts of them from a command-line terminal.
Some examples were already given in the Introduction, but here we will show some additional options.
To see the available options, in a command-line terminal, type pydicom help show
or pydicom show -h
.
$ pydicom help show
usage: pydicom show [-h] [-x] [-t] [-q] filespec
Display all or part of a DICOM file
positional arguments:
filespec File specification, in format [pydicom::]filename[::element]. If `pydicom::` prefix is present,
then use the pydicom test file with that name. If `element` is given, use only that data element
within the file. Examples: path/to/your_file.dcm, your_file.dcm::StudyDate,
your_file.dcm::(0001,0001), pydicom::rtplan.dcm::BeamSequence[0],
yourplan.dcm::BeamSequence[0].BeamNumber, pydicom::rtplan.dcm::(300A,00B0)[0].(300A,00B6)
optional arguments:
-h, --help show this help message and exit
-x, --exclude-private
Don't show private data elements
-t, --top Only show top level
-q, --quiet Only show basic information
The basic command with no options shows all data elements and nested sequences:
$ pydicom show pydicom::CT_small.dcm
Dataset.file_meta -------------------------------
(0002, 0000) File Meta Information Group Length UL: 192
(0002, 0001) File Meta Information Version OB: b'\x00\x01'
(0002, 0002) Media Storage SOP Class UID UI: CT Image Storage
(0002, 0003) Media Storage SOP Instance UID UI: 1.3.6.1.4.1.5962.1.1.1.1.1.20040119072730.12322
(0002, 0010) Transfer Syntax UID UI: Explicit VR Little Endian
(0002, 0012) Implementation Class UID UI: 1.3.6.1.4.1.5962.2
(0002, 0013) Implementation Version Name SH: 'DCTOOL100'
(0002, 0016) Source Application Entity Title AE: 'CLUNIE1'
-------------------------------------------------
(0008, 0005) Specific Character Set CS: 'ISO_IR 100'
(0008, 0008) Image Type CS: ['ORIGINAL', 'PRIMARY', 'AXIAL']
(0008, 0012) Instance Creation Date DA: '20040119'
(0008, 0013) Instance Creation Time TM: '072731'
(0008, 0014) Instance Creator UID UI: 1.3.6.1.4.1.5962.3
(0008, 0016) SOP Class UID UI: CT Image Storage
(0008, 0018) SOP Instance UID UI: 1.3.6.1.4.1.5962.1.1.1.1.1.20040119072730.12322
(0008, 0020) Study Date DA: '20040119'
.
.
.
(0043, 104b) [DAS xm pattern] SL: 0
(0043, 104c) [TGGC trigger mode] SS: 0
(0043, 104d) [Start scan to X-ray on delay] FL: 0.0
(0043, 104e) [Duration of X-ray on] FL: 10.60060977935791
(7fe0, 0010) Pixel Data OW: Array of 32768 elements
(fffc, fffc) Data Set Trailing Padding OB: Array of 126 elements
Note that prefixing the file specification with pydicom::
will read the file
from the pydicom test data files rather than from the file system.
You can also show just parts of the DICOM file by specifying a data element using the usual pydicom keyword notation:
$ pydicom show pydicom::CT_small.dcm::PatientName
CompressedSamples^CT1
$ pydicom show pydicom::rtplan.dcm::FractionGroupSequence
[(300a, 0071) Fraction Group Number IS: "1"
(300a, 0078) Number of Fractions Planned IS: "30"
(300a, 0080) Number of Beams IS: "1"
(300a, 00a0) Number of Brachy Application Setups IS: "0"
(300c, 0004) Referenced Beam Sequence 1 item(s) ----
(300a, 0082) Beam Dose Specification Point DS: [239.531250000000, 239.531250000000, -751.87000000000]
(300a, 0084) Beam Dose DS: "1.0275401"
(300a, 0086) Beam Meterset DS: "116.0036697"
(300c, 0006) Referenced Beam Number IS: "1"
---------]
You can also use a tag number in format (group,elem) anywhere a DICOM keyword can be used:
$ pydicom show pydicom::ct_small.dcm::(0043,1013)
[107, 21, 4, 2, 20]
$ pydicom show pydicom::rtplan.dcm::(300A,00B0)[0].(300a,0111)
[(300A,0112) Control Point Index IS: '0'
(300A,0114) Nominal Beam Energy DS: '6.00000000000000'
...
The -q
quiet argument shows a minimal version of some of the information in the
file, using just the DICOM keyword and value (not showing the tag numbers
and VR). The example below shows the quiet mode with an image slice:
$ pydicom show -q pydicom::ct_small.dcm
SOPClassUID: CT Image Storage
PatientName: CompressedSamples^CT1
PatientID: 1CT1
StudyID: 1CT1
StudyDate: 20040119
StudyTime: 072730
StudyDescription: e+1
BitsStored: 16
Modality: CT
Rows: 128
Columns: 128
SliceLocation: -77.2040634155
And the following example shows an RT Plan in quiet mode:
$ pydicom show -q pydicom::rtplan.dcm
SOPClassUID: RT Plan Storage
PatientName: Last^First^mid^pre
PatientID: id00001
StudyID: study1
StudyDate: 20030716
StudyTime: 153557
StudyDescription: N/A
Plan Label: Plan1 Plan Name: Plan1
Fraction Group 1 30 fraction(s) planned
Brachy Application Setups: 0
Beam 1 Dose 1.02754010000000 Meterset 116.003669700000
Beam 1 'Field 1' TREATMENT STATIC PHOTON energy 6.00000000000000 gantry 0.0, coll 0.0, couch 0.0 (0 wedges, 0 comps, 0 boli, 0 blocks)
Quiet modes always show the SOP Class UID, patient and study information as shown in the above two examples. After those elements, custom values for different SOP classes are shown. Currently “Image Storage” and “RT Plan Storage” classes have custom extra information. Please submit an issue on the pydicom issues list or a pull request to help us expand the list of custom ‘quiet’ mode SOP Classes.
pydicom codify
command¶
The pydicom codify
command takes a DICOM file and produces Python code to
recreate that file, or, optionally a subset within that file.
See Using codify for full details of writing a complete file. Here we will review the command-line options in more detail than in that section, and show how to export a dataset within a DICOM file that has sequences.
Warning
The code produced by codify
will contain all the information in the original
file, which may include private health information or other sensitive
information.
A simple example¶
A simple example of using the codify
command would be:
$ pydicom codify pydicom::rtplan.dcm
# Coded version of DICOM file 'C:\git\pydicom\pydicom\data\test_files\rtplan.dcm'
# Produced by pydicom codify utility script
import pydicom
from pydicom.dataset import Dataset, FileMetaDataset
from pydicom.sequence import Sequence
# Main data elements
ds = Dataset()
ds.InstanceCreationDate = '20030903'
ds.InstanceCreationTime = '150031'
ds.SOPClassUID = '1.2.840.10008.5.1.4.1.1.481.5'
ds.SOPInstanceUID = '1.2.777.777.77.7.7777.7777.20030903150023'
ds.StudyDate = '20030716'
ds.StudyTime = '153557'
.
.
.
Note that prefixing the file specification with pydicom::
will read the file
from the pydicom test data files rather than from the file system.
Command options¶
In the above example, the output was directed to screen, because no output file
was specified. To see the available command options, use the help
command:
pydicom help codify
usage: pydicom codify [-h] [-e EXCLUDE_SIZE] [-p] [-s SAVE_AS] filespec [outfile]
Read a DICOM file and produce the *pydicom* (Python) code which can create that file
positional arguments:
filespec File specification, in format [pydicom::]filename[::element]. If `pydicom::` prefix is present,
then use the pydicom test file with that name. If `element` is given, use only that data element
within the file. Examples: path/to/your_file.dcm, your_file.dcm::StudyDate,
your_file.dcm::(0001,0001), pydicom::rtplan.dcm::BeamSequence[0],
yourplan.dcm::BeamSequence[0].BeamNumber, pydicom::rtplan.dcm::(300A,00B0)[0].(300A,00B6)
outfile Filename to write python code to. If not specified, code is written to
stdout
optional arguments:
-h, --help show this help message and exit
-e EXCLUDE_SIZE, --exclude-size EXCLUDE_SIZE
Exclude binary data larger than specified (bytes). Default is 100 bytes
-p, --include-private
Include private data elements (default is to exclude them)
-s SAVE_AS, --save-as SAVE_AS
Specify the filename for ds.save_as(save_filename); otherwise the input name
+ '_from_codify' will be used
Binary data (e.g. pixels) larger than --exclude-size (default 100 bytes) is not included. A dummy
line with a syntax error is produced. Private data elements are not included by default.
For example:
pydicom codify -s savename.dcm dicomfile.dcm pythoncode.py
would read the DICOM file “dicomfile.dcm” and write the Python code
to file “pythoncode.py”. In that code, near the end of the file
would be a ds.save_as("savename.dcm", ...)
line.
Note
By default, any private data elements within the file are not translated
to code. If you want to include them, use the -p
parameter.
Codifying a part of a DICOM file¶
Note that the filespec
argument to the codify
command, as for
the show command, allows you to specify a data element within the file,
rather than the whole file:
pydicom codify pydicom::rtplan.dcm::FractionGroupSequence[0]
# Coded version of non-file dataset
...
# Main data elements
ds = Dataset()
ds.FractionGroupNumber = "1"
ds.NumberOfFractionsPlanned = "30"
ds.NumberOfBeams = "1"
ds.NumberOfBrachyApplicationSetups = "0"
# Referenced Beam Sequence
refd_beam_sequence = Sequence()
ds.ReferencedBeamSequence = refd_beam_sequence
# Referenced Beam Sequence: Referenced Beam 1
refd_beam1 = Dataset()
refd_beam1.BeamDoseSpecificationPoint = [239.531250000000, 239.531250000000, -751.87000000000]
...
Currently, only a data element which is a Dataset
(an item within a Sequence
) is accepted.
The resulting code would not on its own produce a correct DICOM file,
but could be useful as a model when creating
more complete code. For example, issuing code for one item in a
Sequence
could be the starting point towards a loop
producing a number of sequence items.
Extending the CLI¶
Developers can create their own ‘subcommands’ for the pydicom
command,
by adding entry points to their package’s setup.py file, specifying a callback
function to register the subcommand and its arguments.
If you wanted to create two subcommands, ‘command1’ and ‘command2’, your setup.py file should include something like:
from setuptools import setup
if __name__ == '__main__':
setup(
name="yourpackage",
# various setup options...,
entry_points = {
"pydicom_subcommands": [
"command1 = yourpackage.command1module.add_subparser",
"command2 = yourpackage.command2module.add_subparser"
]
}
)
The "pydicom_subcommands"
is a literal string; this must not be
changed or pydicom will not find your subcommand.
The add_subparser
function name could be changed if you wish, but usually
would be used by convention, and is assumed in the following examples.
In the module you have specified, create the add_subparser
function,
which takes a single argument subparsers
, and a do_command
function,
which will take the call when you subcommand is actually used at the command
line:
from pydicom.cli.main import filespec_help, filespec_parser
def add_subparser(subparsers):
# Register the sub-parser
subparser = subparsers.add_parser(
"subcommandname",
description="Summary of your subcommand"
)
subparser.add_argument(
"filespec",
help=filespec_help,
type=filespec_parser
)
subparser.add_argument(
...
)
subparser.set_defaults(func=do_command)
And define your command function:
def do_command(args):
for ds, element_val in args.filespec:
if args.yourarg:
# Do something...
# work with the dataset ds or element as needed...
The pydicom
command uses Python’s
argparse library to
process commands.
The above code snippets show adding the filespec
argument, and processing
the resulting dataset-element_value pairs in the do_command()
function. This is
recommended if you wish to use the filespec as was seen in the pydicom show command
and pydicom codify command sections. If not, you can just create a normal
arg with the type set to argparse.FileType
to open files yourself.
The above has been shown in relation to a different package than pydicom;
however, if you think your command has general use, please consider contributing
it to pydicom: in that case, change the entry points in the pydicom
setup.py
script, and add a module under pydicom.cli
and create a
pull request.