Skip to main content

Karoo Array Telescope data access library for interacting with data sets in the MeerKAT Visibility Format (MVF)

Project description

katdal

This package serves as a data access library to interact with the chunk stores and HDF5 files produced by the MeerKAT radio telescope and its predecessors (KAT-7 and Fringe Finder), which are collectively known as MeerKAT Visibility Format (MVF) data sets. It uses memory carefully, allowing data sets to be inspected and partially loaded into memory. Data sets may be concatenated and split via a flexible selection mechanism. In addition, it provides a script to convert these data sets to CASA MeasurementSets.

Quick Tutorial

Open any data set through a single function to obtain a data set object:

import katdal
d = katdal.open('1234567890.h5')

The open function automatically determines the version and storage location of the data set. The versions roughly map to the various instruments:

- v1 : Fringe Finder (HDF5 file)
- v2 : KAT-7 (HDF5 file)
- v3 : MeerKAT (HDF5 file)
- v4 : MeerKAT (RDB file + chunk store based on objects in Ceph)

Each MVFv4 data set is split into a Redis dump (aka RDB) file containing the metadata in the form of a telescope state database, and a chunk store containing the visibility data split into many small blocks or chunks (typically served by a Ceph object store over the network). The RDB file is the main entry point to the data set and it can be accessed directly from the MeerKAT SDP archive if you have the appropriate permissions:

# This is just for illustration - the real URL looks a bit different
d = katdal.open('https://archive/1234567890/1234567890_sdp_l0.rdb?token=AsD3')

Multiple data sets (even of different versions) may also be concatenated together (as long as they have the same dump rate):

d = katdal.open(['1234567890.h5', '1234567891.h5'])

Inspect the contents of the data set by printing the object:

print(d)

Here is a typical output:

===============================================================================
Name: 1313067732.h5 (version 2.0)
===============================================================================
Observer: someone  Experiment ID: 2118d346-c41a-11e0-b2df-a4badb44fe9f
Description: 'Track on Hyd A,Vir A, 3C 286 and 3C 273'
Observed from 2011-08-11 15:02:14.072 SAST to 2011-08-11 15:19:47.810 SAST
Dump rate: 1.00025 Hz
Subarrays: 1
ID  Antennas                            Inputs  Corrprods
 0  ant1,ant2,ant3,ant4,ant5,ant6,ant7  14      112
Spectral Windows: 1
ID  CentreFreq(MHz)  Bandwidth(MHz)  Channels  ChannelWidth(kHz)
 0  1822.000         400.000          1024      390.625
-------------------------------------------------------------------------------
Data selected according to the following criteria:
subarray=0
ants=['ant1', 'ant2', 'ant3', 'ant4', 'ant5', 'ant6', 'ant7']
spw=0
-------------------------------------------------------------------------------
Shape: (1054 dumps, 1024 channels, 112 correlation products) => Size: 967.049 MB
Antennas: *ant1,ant2,ant3,ant4,ant5,ant6,ant7  Inputs: 14  Autocorr: yes  Crosscorr: yes
Channels: 1024 (index 0 - 1023, 2021.805 MHz - 1622.195 MHz), each 390.625 kHz wide
Targets: 4 selected out of 4 in catalogue
ID  Name    Type      RA(J2000)     DEC(J2000)  Tags  Dumps  ModelFlux(Jy)
 0  Hyd A   radec      9:18:05.28  -12:05:48.9          333      33.63
 1  Vir A   radec     12:30:49.42   12:23:28.0          251     166.50
 2  3C 286  radec     13:31:08.29   30:30:33.0          230      12.97
 3  3C 273  radec     12:29:06.70    2:03:08.6          240      39.96
Scans: 8 selected out of 8 total       Compscans: 1 selected out of 1 total
Date        Timerange(UTC)       ScanState  CompScanLabel  Dumps  Target
11-Aug-2011/13:02:14 - 13:04:26    0:slew     0:             133    0:Hyd A
            13:04:27 - 13:07:46    1:track    0:             200    0:Hyd A
            13:07:47 - 13:08:37    2:slew     0:              51    1:Vir A
            13:08:38 - 13:11:57    3:track    0:             200    1:Vir A
            13:11:58 - 13:12:27    4:slew     0:              30    2:3C 286
            13:12:28 - 13:15:47    5:track    0:             200    2:3C 286
            13:15:48 - 13:16:27    6:slew     0:              40    3:3C 273
            13:16:28 - 13:19:47    7:track    0:             200    3:3C 273

The first segment of the printout displays the static information of the data set, including observer, dump rate and all the available subarrays and spectral windows in the data set. The second segment (between the dashed lines) highlights the active selection criteria. The last segment displays dynamic information that is influenced by the selection, including the overall visibility array shape, antennas, channel frequencies, targets and scan info.

The data set is built around the concept of a three-dimensional visibility array with dimensions of time, frequency and correlation product. This is reflected in the shape of the dataset:

d.shape

which returns (1054, 1024, 112), meaning 1054 dumps by 1024 channels by 112 correlation products.

Let’s select a subset of the data set:

d.select(scans='track', channels=slice(200, 300), ants='ant4')
print(d)

This results in the following printout:

===============================================================================
Name: /Users/schwardt/Downloads/1313067732.h5 (version 2.0)
===============================================================================
Observer: siphelele  Experiment ID: 2118d346-c41a-11e0-b2df-a4badb44fe9f
Description: 'track on Hyd A,Vir A, 3C 286 and 3C 273 for Lud'
Observed from 2011-08-11 15:02:14.072 SAST to 2011-08-11 15:19:47.810 SAST
Dump rate: 1.00025 Hz
Subarrays: 1
ID  Antennas                            Inputs  Corrprods
 0  ant1,ant2,ant3,ant4,ant5,ant6,ant7  14      112
Spectral Windows: 1
ID  CentreFreq(MHz)  Bandwidth(MHz)  Channels  ChannelWidth(kHz)
 0  1822.000         400.000          1024      390.625
-------------------------------------------------------------------------------
Data selected according to the following criteria:
channels=slice(200, 300, None)
subarray=0
scans='track'
ants='ant4'
spw=0
-------------------------------------------------------------------------------
Shape: (800 dumps, 100 channels, 4 correlation products) => Size: 2.560 MB
Antennas: ant4  Inputs: 2  Autocorr: yes  Crosscorr: no
Channels: 100 (index 200 - 299, 1943.680 MHz - 1905.008 MHz), each 390.625 kHz wide
Targets: 4 selected out of 4 in catalogue
ID  Name    Type      RA(J2000)     DEC(J2000)  Tags  Dumps  ModelFlux(Jy)
 0  Hyd A   radec      9:18:05.28  -12:05:48.9          200      31.83
 1  Vir A   radec     12:30:49.42   12:23:28.0          200     159.06
 2  3C 286  radec     13:31:08.29   30:30:33.0          200      12.61
 3  3C 273  radec     12:29:06.70    2:03:08.6          200      39.32
Scans: 4 selected out of 8 total       Compscans: 1 selected out of 1 total
Date        Timerange(UTC)       ScanState  CompScanLabel  Dumps  Target
11-Aug-2011/13:04:27 - 13:07:46    1:track    0:             200    0:Hyd A
            13:08:38 - 13:11:57    3:track    0:             200    1:Vir A
            13:12:28 - 13:15:47    5:track    0:             200    2:3C 286
            13:16:28 - 13:19:47    7:track    0:             200    3:3C 273

Compared to the first printout, the static information has remained the same while the dynamic information now reflects the selected subset. There are many possible selection criteria, as illustrated below:

d.select(timerange=('2011-08-11 13:10:00', '2011-08-11 13:15:00'), targets=[1, 2])
d.select(spw=0, subarray=0)
d.select(ants='ant1,ant2', pol='H', scans=(0,1,2), freqrange=(1700e6, 1800e6))

See the docstring of DataSet.select for more detailed information (i.e. do d.select? in IPython). Take note that only one subarray and one spectral window must be selected.

Once a subset of the data has been selected, you can access the data and timestamps on the data set object:

vis = d.vis[:]
timestamps = d.timestamps[:]

Note the [:] indexing, as the vis and timestamps properties are special LazyIndexer objects that only give you the actual data when you use indexing, in order not to inadvertently load the entire array into memory.

For the example dataset and no selection the vis array will have a shape of (1054, 1024, 112). The time dimension is labelled by d.timestamps, the frequency dimension by d.channel_freqs and the correlation product dimension by d.corr_products.

Another key concept in the data set object is that of sensors. These are named time series of arbitrary data that are either loaded from the data set (actual sensors) or calculated on the fly (virtual sensors). Both variants are accessed through the sensor cache (available as d.sensor) and cached there after the first access. The data set object also provides convenient properties to expose commonly-used sensors, as shown in the plot example below:

import matplotlib.pyplot as plt
plt.plot(d.az, d.el, 'o')
plt.xlabel('Azimuth (degrees)')
plt.ylabel('Elevation (degrees)')

Other useful attributes include ra, dec, lst, mjd, u, v, w, target_x and target_y. These are all one-dimensional NumPy arrays that dynamically change length depending on the active selection.

As in katdal’s predecessor (scape) there is a DataSet.scans generator that allows you to step through the scans in the data set. It returns the scan index, scan state and target object on each iteration, and updates the active selection on the data set to include only the current scan. It is also possible to iterate through the compound scans with the DataSet.compscans generator, which yields the compound scan index, label and first target on each iteration for convenience. These two iterators may also be used together to traverse the data set structure:

for compscan, label, target in d.compscans():
    plt.figure()
    for scan, state, target in d.scans():
        if state in ('scan', 'track'):
            plt.plot(d.ra, d.dec, 'o')
    plt.xlabel('Right ascension (J2000 degrees)')
    plt.ylabel('Declination (J2000 degrees)')
    plt.title(target.name)

Finally, all the targets (or fields) in the data set are stored in a catalogue available at d.catalogue, and the original HDF5 file is still accessible via a back door installed at d.file in the case of a single-file data set (v3 or older). On a v4 data set, d.source provides access to the underlying telstate for metadata and the chunk store for data.

History

0.19 (2021-11-23)

  • Support scans and non-radec targets like planets in mvftoms (#333)

  • Expose the raw flags of MVF4 datasets (#335)

  • Expose CBF F-engine sensors: applied delays, phases and gains (#338)

  • Verify that S3 bucket is not empty to detect datasets archived to tape (#344)

  • Populate SIGMA_SPECTRUM and redo SIGMA and WEIGHT in mvftoms (#347)

  • Have a sensible DataSet.name and also add a separate DataSet.url (#337)

  • Allow deselection of antennas using ‘~m0XX’ (#340)

  • Allow nested DaskLazyIndexers (#336)

  • Fix mvftoms on macOS and Python 3.8+ (#339)

0.18 (2021-04-20)

  • Switch to PyJWT 2 and Python 3.6, cleaning up Python 2 relics (#321 - #323)

  • Allow preselection of channels and dumps upon katdal.open() to save time and memory (#324)

  • Allow user to select fields, scans and antennas in mvftoms (#269)

  • Support h5py 3.0 string handling in MVF3 (#331)

  • Refactor requirement files to remove recursive dependencies (#329)

0.17 (2021-01-27)

  • This is the last release that will support Python 3.5

  • Pin PyJWT version to 1.x to avoid breaking API changes (#320)

  • Van Vleck correction! (autocorrelations only, though) (#316)

  • Expose excision, aka raw weights (#308)

  • Better unit testing of DataSource and S3ChunkStore in general (#319)

  • Support indexed telstate keys (the 1000th cut that killed Python 2) (#304)

  • Split out separate utility classes for Minio (#310)

  • Fix filtering of sensor events with invalid status (#306)

0.16 (2020-08-28)

  • This is the last release that will support Python 2 (python2 maintenance branch)

  • New ‘time_offset’ sensor property that adjusts timestamps of any sensor (#307)

  • Fix calculation of cbf_dump_period for ‘wide’ / ‘narrowN’ instruments (#301)

  • Increase katstore search window by 600 seconds to find infrequent updates (#302)

  • Refactor SensorData to become a lazy abstract interface without caching (#292)

  • Refactor SensorCache to use MutableMapping (#300)

  • Fix rx_serial sensor use and file mode warning in MVFv3 files (#298, #299)

0.15 (2020-03-13)

  • Improve S3 chunk store: check tokens, improve timeouts and retries (#272 - #277)

  • Retry truncated reads and 50x errors due to S3 server overload (#274)

  • Apply flux calibration if available (#278, #279)

  • Improve mvf_rechunk and mvf_read_benchmark scripts (#280, #281, #284)

  • Fix selection by target description (#271)

  • Mark Python 2 support as deprecated (#282)

0.14 (2019-10-02)

  • Make L2 product by applying self-calibration corrections (#253 - #256)

  • Speed up uvw calculations (#252, #262)

  • Produce documentation on readthedocs.org (#244, #245, #247, #250, #261)

  • Clean up mvftoms and fix REST_FREQUENCY in SOURCE sub-table (#258)

  • Support katstore64 API (#265)

  • Improve chunk store: detect short reads, speed up handling of lost data (#259, #260)

  • Use katpoint 0.9 and dask 1.2.1 features (#262, #243)

0.13 (2019-05-09)

  • Load RDB files straight from archive (#233, #241)

  • Retrieve raw sensor data from CAM katstore (#234)

  • Work around one-CBF-dump offset issue (#238)

  • Improved MS output: fixed RECEPTOR_ANGLE (#230), added WEIGHT_SPECTRUM (#231)

  • Various optimisations to applycal (#224), weights (#226), S3 reads (#229)

  • Use katsdptelstate 0.8 and dask 1.1 features (#228, #233, #240)

0.12 (2019-02-12)

  • Optionally make L1 product by applying calibration corrections (#194 - #198)

  • Let default reference antenna in v4 datasets be “array” antenna (#202, #220)

  • Use katsdptelstate v0.7: generic encodings, memory backend (#196, #201, #212)

  • Prepare for multi-dump chunks (#213, #214, #216, #217, #219)

  • Allow L1 flags to be ignored (#209, #210)

  • Deal with deprecated dask features (#204, #215)

  • Remove RADOS chunk store (it’s all via S3 from here on)

0.11 (2018-10-15)

  • Python 3 support via python-future (finally!)

  • Load L1 flags if available (#164)

  • Reduced memory usage (#165) and speedups (#155, #169, #170, #171, #182)

  • S3 chunk store now uses requests directly instead of via botocore (#166)

  • Let lazy indexer use oindex semantics like in the past (#180)

  • Fix concatenated data sets (#161)

  • Fix IPython / Jupyter tab completion for sensor cache (#176)

0.10.1 (2018-05-18)

  • Restore NumPy 1.14 support (all data flagged otherwise)

0.10 (2018-05-17)

  • Rally around the MeerKAT Visibility Format (MVF)

  • First optimised converter from MVF v4 to MS: mvftoms

  • Latest v4 fixes (synthetic timestamps, autodetection, NPY files in Ceph)

  • Flag and zero missing chunks

  • Now requires katsdptelstate (released), dask, h5py 2.3 and Python 2.7

  • Restore S3 unit tests and NumPy 1.11 (on Ubuntu 16.04) support

0.9.5 (2018-02-22)

  • New HDF5 v3.9 file format in anticipation of v4 (affects obs_params)

  • Fix receiver serial numbers in recent MeerKAT data sets

  • Add dask support to ChunkStore

  • katdal.open() works on v4 RDB files

0.9 (2018-01-16)

  • New ChunkStore and telstate-based parser for future v4 format

  • Use python-casacore (>=2.2.1) to create Measurement Sets instead of blank.ms

  • Read new-style noise diode sensor names, serial numbers and L0 stream metadata

  • Select multiple polarisations (useful for cross-pol)

  • Relax the “expected number of dumps” check to avoid spurious warnings

  • Fix NumPy 1.14 warnings

0.8 (2017-08-08)

  • Fix upside-down MeerKAT images

  • SensorData rework to load gain solutions and access telstate efficiently

  • Improve mapping of sensor events onto dumps, especially for long (8 s) dumps

  • Fix NumPy 1.13 warnings and errors

  • Support UHF receivers

0.7.1 (2017-01-19)

  • Fix MODEL_DATA / CORRECTED_DATA shapes in h5toms

  • Produce calibration solution tables in h5toms and improve error messages

  • Autodetect receiver band on older RTS files

0.7 (2016-12-14)

  • Support weights in file and improve vis / weights / flags API

  • Support multiple receivers and improve centre frequency extraction

  • Speed up h5toms by ordering visibilities by time

  • Fix band selection and corr products for latest SDP (cam2telstate)

  • Allow explicit MS names in h5toms

0.6 (2016-09-16)

  • Initial release of katdal

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

katdal-0.19.tar.gz (242.4 kB view hashes)

Uploaded Source

Built Distribution

katdal-0.19-py3-none-any.whl (236.1 kB view hashes)

Uploaded Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page