1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
| Master | Development | Anaconda binaries |
|--------|-------------|-------------------|
| [![Build Status](https://anvil.softeng-support.ac.uk/jenkins/buildStatus/icon?job=CILsingle/CCPi-Framework)](https://anvil.softeng-support.ac.uk/jenkins/job/CILsingle/job/CCPi-Framework/) | [![Build Status](https://anvil.softeng-support.ac.uk/jenkins/buildStatus/icon?job=CILsingle/CCPi-Framework-dev)](https://anvil.softeng-support.ac.uk/jenkins/job/CILsingle/job/CCPi-Framework-dev/) |![conda version](https://anaconda.org/ccpi/ccpi-framework/badges/version.svg) ![conda last release](https://anaconda.org/ccpi/ccpi-framework/badges/latest_release_date.svg) [![conda platforms](https://anaconda.org/ccpi/ccpi-framework/badges/platforms.svg) ![conda dowloads](https://anaconda.org/ccpi/ccpi-framework/badges/downloads.svg)](https://anaconda.org/ccpi/ccpi-framework) |
# CCPi-Framework
Basic Python Framework for CIL
This package provides a common framework, hence the name, for the analysis of data in the CT pipeline and quick development of novel reconstruction algorithms.
## Installation
Binary installation of the CCPi Framework can be done with `conda`. Install a new environment as in the following. Some packages are optional (cvxpy and ccpi-astra)
```bash
conda create -y --name cil python=3.6.7 cvxpy=1.0.15 lapack numpy=1.14 ccpi-framework ccpi-astra tomophantom ccpi-plugins -c oxfordcontrol -c conda-forge -c astra-toolbox -c ccpi/label/dev
```
### Components
Some concepts are so much overlapping with the CCPPETMR [SIRF](https://github.com/CCPPETMR/SIRF) project that we have chosen to stick to their naming and conventions, in the hope that we may be able to complement each other (at least in Python).
This package consists of the following Python modules:
1. `ccpi.framework`
2. `ccpi.optimisation`
3. `ccpi.io`
### `ccpi.framework`
In `ccpi.framework` we define a number of common classes normally used in tomography:
* `DataContainer`
* `DataSetProcessor`
* `ImageData`
* `AcquisitionData`
* `BlockDataContainer`
#### `DataContainer`
Generic class to hold data. Currently the data is currently held in a numpy arrays, but we are currently planning to create a `GPUDataContainer` and `BoostDataContainer` which will hold the data in an array on GPU or in a boost multidimensional array respectively.
The role of the `DataContainer` is to hold the data and metadata as axis labels and geometry.
`ImageData` and `AcquisitionData` are subclasses aimed at containing 3D/4D data and raw data.
`DataContainer` have basic methods to perform algebric operations between each other and/or with numbers. `DataContainer` provide a simple method to produce a `subset` of themselves based on the axis one would like to have. For instance if a `DataContainer` `A` is 3D and its axis represent temperature, width, height, one could create a reordered `DataContainer` by
```python
B = A.subset(['height', 'width','temperature'])
C = A.subset(temperature = 20)
```
#### `DataProcessor`
Defines a generic DataContainer processor, it accepts `DataContainer` as inputs and outputs `DataContainer`.
The aim of this class is to simplify the writing of processing pipelines.
A `DataProcessor` does calculate its output only when needed and can return a copy of its output (if available) when none of its inputs have changed. Normally it is important to overwrite the `process` method and the `__init__` to describe all the parameter that are specific to the processor.
### `ccpi.optimisation`
This package allows rapid prototyping of optimisation-based reconstruction problems,
i.e. defining and solving different optimization problems to enforce different properties
on the reconstructed image.
Firstly, it provides an object-oriented framework for defining mathematical operators and functions
as well a collection of useful example operators and functions. Both smooth and
non-smooth functions can be used.
Further, it provides a number of high-level generic
implementations of optimisation algorithms to solve genericlly formulated
optimisation problems constructed from operator and function objects.
The fundamental components are:
* `Operator`: A class specifying a operator
* `Function`: A class specifying mathematical functions such as a least squares data fidelity.
* `Algorithm`: Implementation of an iterative optimisation algorithm to solve a particular generic optimisation problem. Algorithms are iterable Python object which can be run in a for loop. Can be stopped and warm restarted.
#### `Operator`
The two most important methods are `direct` and `adjoint` methods that describe the result of
applying the operator, and its adjoint respectively, onto a compatible `DataContainer` input.
The output is another `DataContainer` object or subclass hereof. An important
special case is to represent the tomographic forward and backprojection operations.
#### `Function`
A `function` represents a mathematical function of one or more inputs and is intended
to accept `DataContainer`s as input as well as any additional parameters.
Fixed parameters can be passed in during the creation of the `function` object.
The methods of the `function` reflect the properties of it, for example,
if the function represented is differentiable
the `function` should contain a method `gradient` which should return the gradient of the function evaluated at
an input point. If the function is not differentiable but allows a simple proximal operator, the method
`proximal` should return the proxial operator evaluated at an input point. The function value
is evaluated by calling the function itself, e.g. `f(x)` for a `function`
`f` and input point `x`.
#### `Algorithm`
A number of generic algorithm implementations are provided including Gradient Descent CGLS and FISTA. An algorithm
is designed for a particular generic optimisation problem accepts and number of `Function`s and/or
`Operator`s as input to define a specific instance of the generic optimisation problem to be solved.
They are iterable objects which can be run in a `for` loop. The user can provide a stopping criterion different than the default max_iteration. `Algorithm`s provide a courtesy method `run(number_of_iterations, verbose)` which allows the user to easily run a `number_of_iterations` and receive a print to screen.
New algorithms can be easily created by extending the `Algorithm` class. The user is required to implement only 4 methods: `set_up`, `__init__`, `update` and `update_objective`.
* `set_up` and `__init__` are used to configure the algorithm
* `update` is the actual iteration updating the solution
* `update_objective` defines how the objective is calculated.
For example, the implementation of the `update` of the Gradient Descent algorithm to minimise a `Function` will only be:
```python
def update(self):
self.x += -self.rate * self.objective_function.gradient(self.x)
def update_objective(self):
self.loss.append(self.objective_function(self.x))
```
The `Algorithm` provides the infrastructure to continue iteration, to access the values of the objective function in subsequent iterations, the time for each iteration.
#### Examples
Please see the [demos](https://github.com/vais-ral/CIL-Demos) for examples of defining and using operators, functions and algorithms
to specify and solve optimisation-based reconstruction problems.
|