Crate fitsio [−] [src]
fitsio - a thin wrapper around the cfitsio C library.
- File access
- HDU access
- Creating new HDUs
- Header keys
- Reading file data
- Writing file data
- Raw fits file access
This library wraps the low level cfitsio bindings: fitsio-sys and provides a more
native experience for rust users.
The main interface to a fits file is FitsFile. All file manipulation
and reading starts with this class.
File access
Opening a file:
use fitsio::FitsFile; // let filename = ...; let fptr = FitsFile::open(filename).unwrap();
Alternatively a new file can be created on disk with the companion method
create:
use fitsio::FitsFile; // let filename = ...; let fptr = FitsFile::create(filename).open().unwrap();
The create method returns a NewFitsFile, which is an
internal representation of a temporary fits file on disk, before the file is fully created.
This representation has two methods: open and
with_custom_primary. The open
method actually creates the file on disk, but before calling this method, the
with_custom_primary method can be used to add a custom
primary HDU. This is mostly useful for images. Otherwise, a default primary HDU is created. An
example of not adding a custom primary HDU is shown above. Below we see an example of
with_custom_primary:
use fitsio::FitsFile; // let filename = ...; let description = ImageDescription { data_type: ImageType::DOUBLE_IMG, dimensions: &[52, 103], }; let fptr = FitsFile::create(filename) .with_custom_primary(&description) .open() .unwrap();
From this point, the current HDU can be queried and changed, or fits header cards can be read or file contents can be read.
To open a fits file in read/write mode (to allow changes to the file), the
edit must be used. This opens a file which already exists
on disk for editing.
use fitsio::FitsFile; // let filename = ...; let fptr = FitsFile::edit(filename).unwrap();
HDU access
HDU information belongs to the FitsHdu object. HDUs can be fetched by
String/str or integer (0-indexed).
The HduInfo object contains information about the current HDU:
let hdu = fptr.hdu(0).unwrap(); // image HDU if let HduInfo::ImageInfo { shape, .. } = hdu.info { println!("Image is {}-dimensional", shape.len()); println!("Found image with shape {:?}", shape); } // tables if let HduInfo::TableInfo { column_descriptions, num_rows, .. } = hdu.info { println!("Table contains {} rows", num_rows); println!("Table has {} columns", column_descriptions.len()); }
Creating new HDUs
Creating a new image
New fits images are created with the create_image
method. This method requires the extension name, and an
ImageDescription object, which defines the shape and type of
the desired image:
let image_description = ImageDescription { data_type: ImageType::FLOAT_IMG, dimensions: &[100, 100], }; let hdu = fptr.create_image("EXTNAME".to_string(), &image_description).unwrap();
Creating a new table
Similar to creating new images, new tables are created with the
create_table method. This requires an extension
name, and a slice of ColumnDescriptions:
let first_description = ColumnDescription::new("A") .with_type(ColumnDataType::Int) .create().unwrap(); let second_description = ColumnDescription::new("B") .with_type(ColumnDataType::Long) .create().unwrap(); let descriptions = [first_description, second_description]; let hdu = fptr.create_table("EXTNAME".to_string(), &descriptions).unwrap();
Column descriptions
Columns are described with the
ColumnDescription struct. This
encapsulates: the name of the column, and the data format.
The fits specification allows scalar or vector columns, and the data format is described the
ColumnDataDescription struct, which in
turn encapsulates the number of elements per row element (typically 1), the width of the
column (for strings), and the data type, which is one of the
ColumnDataType members
For the common case of a scalar column, a ColumnDataDescription object can be constructed
with the scalar method:
let desc = ColumnDataDescription::scalar(ColumnDataType::Int); assert_eq!(desc.repeat, 1); assert_eq!(desc.width, 1);
Vector columns can be constructed with the vector method:
let desc = ColumnDataDescription::vector(ColumnDataType::Int, 100); assert_eq!(desc.repeat, 100); assert_eq!(desc.width, 1);
These impl From<...> for String such that the traditional fits column description string can
be obtained:
let desc = ColumnDataDescription::scalar(ColumnDataType::Int); assert_eq!(String::from(desc), "1J".to_string());
Copying HDUs to another file
A HDU can be copied to another open file with the copy_to method. This
requires another open FitsFile object to copy to:
hdu.copy_to(&mut src_fptr, &mut dest_fptr).unwrap();
Deleting a HDU
The current HDU can be deleted using the delete method. Note: this method
takes ownership of self, and as such the FitsHdu object cannot be used after
this is called.
// let fptr = FitsFile::open(...).unwrap(); // let hdu = fptr.hdu(0).unwrap(); hdu.delete(&mut fptr).unwrap(); // Cannot use hdu after this
Iterating over the HDUs in a file
The iter method allows for iteration over the HDUs of a fits file.
for hdu in fptr.iter() { // Do something with hdu }
General calling behaviour
All subsequent data acess is performed through the FitsHdu object. Most methods
take the currently open FitsFile as the first parameter.
Header keys
Header keys are read through the read_key function,
and is generic over types that implement the ReadsKey trait:
let int_value: i64 = fptr.hdu(0).unwrap().read_key(&mut fptr, "INTTEST").unwrap(); // Alternatively let int_value = fptr.hdu(0).unwrap().read_key::<i64>(&mut fptr, "INTTEST").unwrap(); // Or let the compiler infer the types (if possible)
Header cards can be written through the method
write_key. It takes a key name and value. See the
WritesKey trait for supported data types.
fptr.hdu(0).unwrap().write_key(&mut fptr, "foo", 1i64).unwrap(); assert_eq!(fptr.hdu(0).unwrap().read_key::<i64>(&mut fptr, "foo").unwrap(), 1i64);
Reading file data
Reading images
Image data can be read through either
read_section which reads contiguous pixels
between a start index and end index, or
read_region which reads rectangular chunks from
the image.
// Read the first 100 pixels let first_row: Vec<i32> = hdu.read_section(&mut fptr, 0, 100).unwrap(); // Read a square section of the image let xcoord = 0..10; let ycoord = 0..10; let chunk: Vec<i32> = hdu.read_region(&mut fptr, &[&ycoord, &xcoord]).unwrap();
Some convenience methods are available for reading rows of the image. This is typically useful as it's an efficient access method:
let start_row = 0; let num_rows = 10; let first_few_rows: Vec<f32> = hdu.read_rows(&mut fptr, start_row, num_rows).unwrap(); // 10 rows of 100 columns assert_eq!(first_few_rows.len(), 1000);
The whole image can also be read into memory:
let image_data: Vec<f32> = hdu.read_image(&mut fptr, ).unwrap(); // 100 rows of 100 columns assert_eq!(image_data.len(), 10_000);
Reading tables
Columns can be read using the read_col function,
which can convert data types on the fly. See the ReadsCol trait for
supported data types.
let integer_data: Vec<i32> = hdu.and_then(|hdu| hdu.read_col(&mut fptr, "intcol")).unwrap();
Iterating over columns
Iterate over the columns with columns.
for column in hdu.columns(&mut fptr) { // Do something with column }
Writing file data
Writing images
Image data is written through two methods on the HDU object:
write_section and
write_region.
write_section requires a start index and
end index and data to write. The data parameter needs to be a slice, meaning any contiguous
memory storage method (e.g. Vec) can be passed.
let data_to_write: Vec<f64> = vec![1.0, 2.0, 3.0]; hdu.write_section(&mut fptr, 0, data_to_write.len(), &data_to_write).unwrap();
write_region takes a slice of ranges with which
the data is to be written, and the data to write.
let data_to_write: Vec<f64> = vec![1.0, 2.0, 3.0, 4.0]; let ranges = [&(0..1), &(0..1)]; hdu.write_region(&mut fptr, &ranges, &data_to_write).unwrap();
Resizing an image
Images can be resized to a new shape using the resize method.
The method takes the open FitsFile, and an slice of usize values. Note:
currently fitsio only supports slices with length 2, i.e. a 2D image.
resize takes ownership self to force the user to fetch the HDU object
again. This ensures the image changes are reflected in the hew HDU object.
hdu.resize(&mut fptr, &[1024, 1024]).unwrap(); // Have to get the HDU again, to reflect the latest changes let hdu = fptr.hdu(0).unwrap(); match hdu.info { HduInfo::ImageInfo { shape, .. } => { assert_eq!(shape, [1024, 1024]); } _ => panic!("Unexpected hdu type"), }
Writing tables
Writing table data
Tablular data can either be written with write_col or
write_col_range.
write_col writes an entire column's worth of data to the file. It does
not check how many rows are in the file, but extends the table if the length of data is longer
than the table length.
let data_to_write: Vec<i32> = vec![10101; 5]; hdu.write_col(&mut fptr, "bar", &data_to_write).unwrap(); let data: Vec<i32> = hdu.read_col(&mut fptr, "bar").unwrap(); assert_eq!(data, vec![10101, 10101, 10101, 10101, 10101]);
write_col_range writes data to a range of rows in a table. The
range is inclusive of both the upper and lower bounds, so 0..4 writes 5 elements.
let data_to_write: Vec<i32> = vec![10101; 10]; hdu.write_col_range(&mut fptr, "bar", &data_to_write, &(0..4)).unwrap(); let data: Vec<i32> = hdu.read_col(&mut fptr, "bar").unwrap(); assert_eq!(data, vec![10101, 10101, 10101, 10101, 10101]);
Inserting columns
Two methods on the HDU object allow for adding new columns:
append_column
and insert_column.
append_column adds a new column as the last column member, and is
generally
preferred as it does not require shifting of data within the file.
let column_description = ColumnDescription::new("abcdefg") .with_type(ColumnDataType::Int) .create().unwrap(); hdu.append_column(&mut fptr, &column_description).unwrap();
Deleting columns
The HDU object has the method delete_column which removes a column.
The column can either be accessed by integer or name
let newhdu = hdu.delete_column(&mut fptr, "bar").unwrap(); // or let newhdu = hdu.delete_column(&mut fptr, 0).unwrap();
Raw fits file access
If this library does not support the particular use case that is needed, the raw fitsfile
pointer can be accessed:
extern crate fitsio_sys; let fptr = FitsFile::open(filename).unwrap(); /* Find out the number of HDUs in the file */ let mut num_hdus = 0; let mut status = 0; unsafe { let fitsfile = fptr.as_raw(); /* Use the unsafe fitsio-sys low level library to call a function that is possibly not implemented in this crate */ fitsio_sys::ffthdu(fitsfile, &mut num_hdus, &mut status); } assert_eq!(num_hdus, 2);
This (unsafe) pointer can then be used with the underlying fitsio-sys library directly.
Reexports
pub use self::fitsfile::FitsFile; |
pub use self::fitsfile::FitsHdu; |
pub use self::types::HduInfo; |
pub use self::errors::Error; |
pub use self::errors::Result; |
Modules
| columndescription |
Handling column descriptions |
| errors |
Errors and error handling |
| fitsfile | |
| types |
Data types used within |