Extension: Json and NetCDF utilities
#include "dg/file/file.h"
|
Serial NetCDF-4 file. More...
Public Types | |
using | Hyperslab = NcHyperslab |
Public Member Functions | |
SerialNcFile ()=default | |
SerialNcFile (const std::filesystem::path &filename, enum NcFileMode mode=nc_nowrite) | |
Open/Create a netCDF file. | |
SerialNcFile (const SerialNcFile &rhs)=delete | |
There can only be exactly one file handle per physical file. | |
SerialNcFile & | operator= (const SerialNcFile &rhs)=delete |
There can only be exactly one file handle per physical file. | |
SerialNcFile (SerialNcFile &&rhs)=default | |
Swap resources between two file handles. | |
SerialNcFile & | operator= (SerialNcFile &&rhs)=default |
Swap resources between two file handles. | |
~SerialNcFile () | |
Close open nc file and release all resources. | |
void | open (const std::filesystem::path &filename, enum NcFileMode mode=nc_nowrite) |
bool | is_open () const noexcept |
void | close () |
Explicitly close a file. | |
void | sync () |
Call nc_sync. | |
int | get_ncid () const noexcept |
Get the ncid of the underlying NetCDF C-API. | |
void | def_grp (std::string name) |
void | def_grp_p (std::filesystem::path path) |
bool | grp_is_defined (std::filesystem::path path) const |
Check for existence of the group given by path. | |
void | set_grp (std::filesystem::path path="") |
Change group to path . | |
void | rename_grp (std::string old_name, std::string new_name) |
int | get_grpid () const noexcept |
Get the NetCDF-C ID of the current group. | |
std::filesystem::path | get_current_path () const |
Get the absolute path of the current group. | |
std::list< std::filesystem::path > | get_grps () const |
std::list< std::filesystem::path > | get_grps_r () const |
Get all subgroups recursively in the current group as absolute paths. | |
void | def_dim (std::string name, size_t size) |
Define a dimension named name of size size . | |
void | rename_dim (std::string old_name, std::string new_name) |
Rename a dimension from old_name to new_name . | |
size_t | get_dim_size (std::string name) const |
Get the size of the dimension named name . | |
std::vector< size_t > | get_dims_shape (const std::vector< std::string > &dims) const |
Get the size of each dimension in dims . | |
std::vector< std::string > | get_dims (bool include_parents=true) const |
std::vector< std::string > | get_unlim_dims () const |
bool | dim_is_defined (std::string name) const |
void | put_att (const std::pair< std::string, nc_att_t > &att, std::string id="") |
Put an individual attribute. | |
template<class S , class T > | |
void | put_att (const std::tuple< S, nc_type, T > &att, std::string id="") |
Put an individual attribute of preset type to variable id. | |
template<class Attributes = std::map<std::string, nc_att_t>> | |
void | put_atts (const Attributes &atts, std::string id="") |
Write a collection of attributes to a NetCDF variable or file. | |
template<class T > | |
T | get_att_as (std::string att_name, std::string id="") const |
Get an attribute named att_name of the group or variable id . | |
template<class T > | |
std::vector< T > | get_att_vec_as (std::string att_name, std::string id="") const |
Short for get_att_as<std::vector<T>>( id, att_name); | |
template<class T > | |
std::map< std::string, T > | get_atts_as (std::string id="") const |
Read all NetCDF attributes of a certain type. | |
std::map< std::string, nc_att_t > | get_atts (std::string id="") const |
void | del_att (std::string att_name, std::string id="") |
Remove an attribute named att_name from variable id . | |
bool | att_is_defined (std::string att_name, std::string id="") const |
void | rename_att (std::string old_att_name, std::string new_att_name, std::string id="") |
template<class T , class Attributes = std::map<std::string, nc_att_t>> | |
void | def_var_as (std::string name, const std::vector< std::string > &dim_names, const Attributes &atts={}) |
Define a variable with given type, dimensions and (optionally) attributes. | |
template<class Attributes = std::map<std::string, nc_att_t>> | |
void | def_var (std::string name, nc_type xtype, const std::vector< std::string > &dim_names, const Attributes &atts={}) |
Define a variable with given type, dimensions and (optionally) attributes. | |
template<class ContainerType , std::enable_if_t< dg::is_vector_v< ContainerType, SharedVectorTag >, bool > = true> | |
void | put_var (std::string name, const NcHyperslab &slab, const ContainerType &data) |
Write data to a variable. | |
template<class ContainerType , class Attributes = std::map<std::string, nc_att_t>, std::enable_if_t< dg::is_vector_v< ContainerType, SharedVectorTag >, bool > = true> | |
void | defput_var (std::string name, const std::vector< std::string > &dim_names, const Attributes &atts, const NcHyperslab &slab, const ContainerType &data) |
Define and put a variable in one go. | |
template<class T , std::enable_if_t< dg::is_scalar_v< T >, bool > = true> | |
void | put_var (std::string name, const std::vector< size_t > &start, T data) |
Write a single data point. | |
template<class T , class Attributes = std::map<std::string, nc_att_t>> | |
void | def_dimvar_as (std::string name, size_t size, const Attributes &atts) |
Define a dimension and dimension variable in one go. | |
template<class ContainerType , class Attributes = std::map<std::string, nc_att_t>> | |
void | defput_dim (std::string name, const Attributes &atts, const ContainerType &abscissas) |
Define a dimension and define and write to a dimension variable in one go. | |
template<class ContainerType , std::enable_if_t< dg::is_vector_v< ContainerType, SharedVectorTag >, bool > = true> | |
void | get_var (std::string name, const NcHyperslab &slab, ContainerType &data) const |
Read hyperslab slab from variable named name into container data . | |
template<class T , std::enable_if_t< dg::is_scalar_v< T >, bool > = true> | |
void | get_var (std::string name, const std::vector< size_t > &start, T &data) const |
Read scalar from position start from variable named name . | |
bool | var_is_defined (std::string name) const |
nc_type | get_var_type (std::string name) const |
std::vector< std::string > | get_var_dims (std::string name) const |
Get the dimension names associated to variable name . | |
std::list< std::string > | get_var_names () const |
Get a list of variable names in the current group. | |
Serial NetCDF-4 file.
Our take on a modern C++ implementation of the NetCDF-4 data model
See here a usage example
|
default |
Construct a File Handle not associated to any file
|
inline |
Open/Create a netCDF file.
filename | Name or path including the name of the netCDF file to open or create. The path may be either absolute or relative to the execution path of the program i.e. relative to std::filesystem::current_path() |
mode | (see NcFileMode for nc_nowrite, nc_write, nc_clobber, nc_noclobber) |
|
delete |
There can only be exactly one file handle per physical file.
The reason is that the destructor releases all resources and thus a copy of the file that is subsequently destroyed leaves the original in an invalid state
|
default |
Swap resources between two file handles.
|
inline |
Close open nc file and release all resources.
std::cerr
if something goes wrong
|
inline |
Check for existence of the attribute named att_name
in variable id
|
inline |
Explicitly close a file.
Closing a file triggers all buffers to be written and memory to be released. After closing a new file can be associated to this handle again.
|
inline |
Define a dimension named name
of size size
.
name | of the dimension. Cannot be the same name as a dimension name already existing in the current group |
size | Size of the dimension to create. Use NC_UNLIMITED to create an unlimited dimension |
def_var
cannot currently use a dimension that is hidden in such a way even though the NetCDF-C API would allow to do it using the dimension ID. In contrast, get_var_dims
may return dimension names that are actually hidden. It is therefore highly recommended to use unique names for all dimensions in the file. Furthermore, notice that while a dimension is visible in subgroups, its associated dimension variable is not.
|
inline |
Define a dimension and dimension variable in one go.
Short for
name | Name of the dimension and associated dimension variable |
size | Size of the dimension to create. Use NC_UNLIMITED to create an unlimited dimension |
atts | Suggested attribute is "axis" : "T" which enable paraview to recognize the dimension as the time axis |
def_var
cannot currently use a dimension that is hidden in such a way even though the NetCDF-C API would allow to do it using the dimension ID. In contrast, get_var_dims
may return dimension names that are actually hidden. It is therefore highly recommended to use unique names for all dimensions in the file. Furthermore, notice that while a dimension is visible in subgroups, its associated dimension variable is not.
|
inline |
Define a group named name
in the current group
Groups can be thought of as directories in a NetCDF-4 file and we therefore use std::filesystem::path
and use bash equivalent operations to manipulate them
Think of this as the bash command mkdir
name
name | of the new group |
|
inline |
Define a group named path
and all required intermediary groups
Groups can be thought of as directories in a NetCDF-4 file and we therefore use std::filesystem::path
and use bash equivalent operations to manipulate them
Think of this as the bash command mkdir
-p path
path | of the new group. Can be absolute or relative to the current group |
|
inline |
Define a variable with given type, dimensions and (optionally) attributes.
name | Name of the variable to define |
xtype | NetCDF typeid |
dim_names | Names of visible dimensions in the current group Can be empty which makes the defined variable a scalar w/o dimensions. |
dg::evaluate
function, which produces vectors where the first dimension of the given grid varies fastest. Thus, when defining variable dimensions the dimension name of the first grid dimension needs to come last. def_var
cannot currently use a dimension that is hidden in such a way even though the NetCDF-C API would allow to do it using the dimension ID. In contrast, get_var_dims
may return dimension names that are actually hidden. It is therefore highly recommended to use unique names for all dimensions in the file. Furthermore, notice that while a dimension is visible in subgroups, its associated dimension variable is not. atts | Attributes to put for the variable |
|
inline |
Define a variable with given type, dimensions and (optionally) attributes.
name | Name of the variable to define |
dim_names | Names of visible dimensions in the current group. Can be empty which makes the defined variable a scalar w/o dimensions. |
dg::evaluate
function, which produces vectors where the first dimension of the given grid varies fastest. Thus, when defining variable dimensions the dimension name of the first grid dimension needs to come last. def_var
cannot currently use a dimension that is hidden in such a way even though the NetCDF-C API would allow to do it using the dimension ID. In contrast, get_var_dims
may return dimension names that are actually hidden. It is therefore highly recommended to use unique names for all dimensions in the file. Furthermore, notice that while a dimension is visible in subgroups, its associated dimension variable is not. T | set the type of the variable |
atts | Attributes to put for the variable |
|
inline |
Define a dimension and define and write to a dimension variable in one go.
name | Name of the dimension and associated dimension variable |
atts | Suggested attributes is for example "axis" : "X" which enable paraview to recognize the dimension as the x axis and "long_name" : "X-coordinate in Cartesian coordinates" |
abscissas | values to write to the dimension variable (the dimension size is inferred from abscissas.size() and the type is inferred from ContainerType |
ContainerType | May be anything that dg::get_tensor_category recognises as a dg::SharedVectorTag . In defput* members the value type determines the NetCDF type of the variable to define |
put*
and get*
members it is possible for data
to have a different value type from the defined value type of the variable that is being written/read, the NetCDF C-API simply converts it to the requested type. For example a variable declared as NC_DOUBLE
can be read as/written from integer or float and vice versa.
|
inline |
Define and put a variable in one go.
Very convenient to "just write" a variable to a file:
Short for
|
inline |
Remove an attribute named att_name
from variable id
.
att_name | Attribute to delete |
id | Variable name in the current group or empty string, in which case the attributes refer to the current group |
|
inline |
Check for existence of the dimension named name
|
inline |
Get an attribute named att_name
of the group or variable id
.
T | Any type in dg::file::nc_att_t or nc_att_t in which case the type specific nc attribute getters are called or std::vector<type> in which case the general nc_get_att is called |
att_name | Name of the attribute |
id | Variable name in the current group or empty string, in which case the attribute refers to the current group |
T
|
inline |
Short for get_att_as<std::vector<T>>( id, att_name);
|
inline |
Short for get_atts_as<nc_att_t>( id)
|
inline |
Read all NetCDF attributes of a certain type.
For example
id | Variable name in the current group or empty string, in which case the attributes refer to the current group |
T | can be a primitive type like int or double or a vector thereof std::vector<int> or a dg::file::nc_att_t in which case attributes of heterogeneous types are captured |
|
inline |
Get the absolute path of the current group.
|
inline |
Get the size of the dimension named name
.
|
inline |
Get all visible dimension names in the current group
The visible dimensions are all the dimensions in the current group and all its parent group.
def_var
cannot currently use a dimension that is hidden in such a way even though the NetCDF-C API would allow to do it using the dimension ID. In contrast, get_var_dims
may return dimension names that are actually hidden. It is therefore highly recommended to use unique names for all dimensions in the file. Furthermore, notice that while a dimension is visible in subgroups, its associated dimension variable is not. include_parents | per default the parent groups will be included in the search for dimensions, if false they are excluded |
|
inline |
Get the size of each dimension in dims
.
|
inlinenoexcept |
Get the NetCDF-C ID of the current group.
|
inline |
Get all subgroups in the current group as absolute paths
Groups can be thought of as directories in a NetCDF-4 file and we therefore use std::filesystem::path
and use bash equivalent operations to manipulate them
|
inline |
Get all subgroups recursively in the current group as absolute paths.
Think of this as ls
-R
Groups can be thought of as directories in a NetCDF-4 file and we therefore use std::filesystem::path
and use bash equivalent operations to manipulate them
|
inlinenoexcept |
Get the ncid
of the underlying NetCDF C-API.
Just if for whatever reason you want to call a NetCDF C-function yourself ... just don't use it for something nasty, like closing the file or whatever
|
inline |
Get all visible unlimited dimension names in the current group
def_var
cannot currently use a dimension that is hidden in such a way even though the NetCDF-C API would allow to do it using the dimension ID. In contrast, get_var_dims
may return dimension names that are actually hidden. It is therefore highly recommended to use unique names for all dimensions in the file. Furthermore, notice that while a dimension is visible in subgroups, its associated dimension variable is not.
|
inline |
Read hyperslab slab
from variable named name
into container data
.
name | of previously defined variable |
slab | Hyperslab to read |
data | Result on output (will be resized to fit hyperslab) |
ContainerType | May be anything that dg::get_tensor_category recognises as a dg::SharedVectorTag . In defput* members the value type determines the NetCDF type of the variable to define |
put*
and get*
members it is possible for data
to have a different value type from the defined value type of the variable that is being written/read, the NetCDF C-API simply converts it to the requested type. For example a variable declared as NC_DOUBLE
can be read as/written from integer or float and vice versa.
|
inline |
Read scalar from position start
from variable named name
.
name | of previously defined variable |
start | coordinate to take scalar from (can be empty for scalar variable) |
data | Result on output |
|
inline |
Get the dimension names associated to variable name
.
name | of the variable |
def_var
cannot currently use a dimension that is hidden in such a way even though the NetCDF-C API would allow to do it using the dimension ID. In contrast, get_var_dims
may return dimension names that are actually hidden. It is therefore highly recommended to use unique names for all dimensions in the file. Furthermore, notice that while a dimension is visible in subgroups, its associated dimension variable is not.
|
inline |
Get a list of variable names in the current group.
We use std::list
here because of how easy it is to sort or filter elemenets. For example
|
inline |
Get the NetCDF typeid of the variable named name
|
inline |
Check for existence of the group given by path.
Groups can be thought of as directories in a NetCDF-4 file and we therefore use std::filesystem::path
and use bash equivalent operations to manipulate them
path | Absolute or relative path to the current group |
|
inlinenoexcept |
Check if a file is associated (i.e. it is open)
|
inline |
Explicitly open or create a netCDF file
filename | Name or path including the name of the netCDF file to open or create. The path may be either absolute or relative to the execution path of the program i.e. relative to std::filesystem::current_path() |
mode | (see NcFileMode for nc_nowrite, nc_write, nc_clobber, nc_noclobber) |
|
delete |
There can only be exactly one file handle per physical file.
The reason is that the destructor releases all resources and thus a copy of the file that is subsequently destroyed leaves the original in an invalid state
|
default |
Swap resources between two file handles.
|
inline |
Put an individual attribute.
att | Attribute consisting of name and value |
id | Variable name in the current group or empty string, in which case the attribute refers to the current group |
att_is_defined
for existence if this is a concern
|
inline |
Put an individual attribute of preset type to variable id.
S | std::string or const char* |
T | Cannot be an nc_att_t |
att | Attribute consisting of name, type and value |
id | Variable name in the current group or empty string, in which case the attribute refers to the current group |
att_is_defined
for existence if this is a concern
|
inline |
Write a collection of attributes to a NetCDF variable or file.
Example code
Attributes | Any Iterable whose values can be used in put_att i.e. either a std::pair or std::tuple |
atts | An iterable containing all the attributes for the variable or file. atts can be empty in which case no attribute is written. |
id | Variable name in the current group or empty string, in which case the attributes refer to the current group |
att_is_defined
for existence if this is a concern
|
inline |
Write data to a variable.
name | Name of the variable to write data to. Must be visible in the current group |
slab | Define where the data is written. The dimension of the slab slab.ndim() must match the number of dimensions of the variable |
data | to write. Size must be at least that of the slab |
ContainerType | May be anything that dg::get_tensor_category recognises as a dg::SharedVectorTag . In defput* members the value type determines the NetCDF type of the variable to define |
put*
and get*
members it is possible for data
to have a different value type from the defined value type of the variable that is being written/read, the NetCDF C-API simply converts it to the requested type. For example a variable declared as NC_DOUBLE
can be read as/written from integer or float and vice versa.
|
inline |
Write a single data point.
name | Name of the variable to write data to. Must be visible in the current group |
start | The coordinates (one for each dimension) to which to write data to |
data | to write |
T | must be convertible to the datatype of the variable name |
|
inline |
Rename an attribute
|
inline |
Rename a dimension from old_name
to new_name
.
|
inline |
rename a subgroup in the current group from old_name
to new_name
Groups can be thought of as directories in a NetCDF-4 file and we therefore use std::filesystem::path
and use bash equivalent operations to manipulate them
|
inline |
Change group to path
.
Groups can be thought of as directories in a NetCDF-4 file and we therefore use std::filesystem::path
and use bash equivalent operations to manipulate them
All subsequent calls to atts, dims and vars are made to that group
path | can be absolute or relative to the current group. Empty string or "/" goes back to root group. "." is the current group and returns immediately. |
|
inline |
Call nc_sync.
|
inline |
Check if variable named name
is defined in the current group