7#include "../dg/backend/memory.h"
138 open( filename, mode);
167 std::cerr << e.
what() << std::endl;
185 void open(
const std::filesystem::path& filename,
197 err = nc_open( filename.string().c_str(), NC_NETCDF4 |
198 NC_NOWRITE, &m_ncid);
201 err = nc_open( filename.string().c_str(), NC_NETCDF4 |
205 err = nc_create( filename.string().c_str(), NC_NETCDF4 |
206 NC_NOCLOBBER, &m_ncid);
209 err = nc_create( filename.string().c_str(), NC_NETCDF4 |
210 NC_CLOBBER, &m_ncid);
239 err = nc_close(m_ncid);
251 err = nc_sync( m_ncid);
275 err = nc_def_grp( m_grp, name.c_str(), &new_grp_id);
292 if( not path.has_root_path())
295 path = (current / path);
297 int groupid = m_ncid;
298 auto rel_path = path.relative_path();
300 for(
auto it = rel_path.begin(); it != rel_path.end(); it++)
302 std::string grp = *it;
304 int retval = nc_inq_ncid( groupid, grp.c_str(), &new_grpid);
305 if( retval != NC_NOERR)
306 err = nc_def_grp( groupid, grp.c_str(), &groupid);
318 std::string name = path.generic_string();
319 if( not path.has_root_path())
322 name = (current / path ).generic_string();
325 int retval = nc_inq_grp_full_ncid( m_ncid, name.c_str(), &grpid);
326 return retval == NC_NOERR;
337 void set_grp( std::filesystem::path path =
"")
341 std::string name = path.generic_string();
344 if ( name ==
"" or name ==
"/")
348 else if ( name ==
"..")
352 err = nc_inq_grp_parent( m_grp, &m_grp);
356 if( not path.has_root_path())
359 name = (current / path ).generic_string();
361 err = nc_inq_grp_full_ncid( m_ncid, name.c_str(), &m_grp);
371 err = nc_inq_grp_ncid( m_grp, old_name.c_str(), &old_grp);
372 err = nc_rename_grp( old_grp, new_name.c_str());
382 return get_grp_path( m_grp);
390 auto grps = get_grps_abs(m_grp);
391 std::list<std::filesystem::path> grps_v;
392 for(
auto grp : grps)
393 grps_v.push_back( grp.second);
409 auto grps = get_grps_abs_r(m_grp);
410 std::list<std::filesystem::path> grps_v;
411 for(
auto grp : grps)
412 grps_v.push_back( grp.second);
435 err = nc_def_dim( m_grp, name.c_str(), size, &dim_id);
443 err = nc_inq_dimid( m_grp, old_name.c_str(), &dimid);
444 err = nc_rename_dim( m_grp, dimid, new_name.c_str());
458 err = nc_inq_dimid( m_grp, name.c_str(), &dimid);
460 err = nc_inq_dimlen( m_grp, dimid, &len);
468 std::vector<size_t> shape( dims.size());
469 for(
unsigned u=0; u<dims.size(); u++)
483 std::vector<std::string>
get_dims(
bool include_parents =
true)
const
490 err = nc_inq_dimids( m_grp, &ndims, NULL, include_parents);
494 err = nc_inq_dimids( m_grp, &ndims, dimids, include_parents);
497 std::vector<std::string> dims;
498 for (
int i = 0; i < ndims; i++)
500 char dimname [ NC_MAX_NAME+1];
501 err = nc_inq_dimname( m_grp, dimids[i], dimname);
502 if( std::find(dims.begin(), dims.end(), std::string(dimname) ) ==
504 dims.push_back(dimname);
520 err = nc_inq_unlimdims( m_grp, &ndims, NULL);
526 err = nc_inq_unlimdims( m_grp, &ndims, dimids);
527 std::vector<std::string> dims;
528 for(
int i=0; i<ndims; i++)
530 char dimname [ NC_MAX_NAME+1];
531 err = nc_inq_dimname( m_grp, dimids[i], dimname);
532 if( std::find(dims.begin(), dims.end(), std::string(dimname) ) ==
534 dims.push_back(dimname);
544 int retval = nc_inq_dimid( m_grp, name.c_str(), &dimid);
545 return retval == NC_NOERR;
555 void put_att(
const std::pair<std::string, nc_att_t>& att, std::string
id =
"")
557 int varid = name2varid(
id);
558 dg::file::detail::put_att( m_grp, varid, att);
570 template<
class S,
class T>
571 void put_att(
const std::tuple<S,nc_type, T>& att, std::string
id =
"")
573 int varid = name2varid(
id);
574 dg::file::detail::put_att( m_grp, varid, att);
590 template<
class Attributes = std::map<std::
string, nc_att_t> >
591 void put_atts(
const Attributes& atts, std::string
id =
"")
593 int varid = name2varid(
id);
594 dg::file::detail::put_atts( m_grp, varid, atts);
611 T
get_att_as(std::string att_name, std::string
id =
"")
const
613 int varid = name2varid(
id);
614 return dg::file::detail::get_att_as<T>( m_grp, varid, att_name);
639 int varid = name2varid(
id);
640 return dg::file::detail::get_atts_as<T>( m_grp, varid);
645 std::map<std::string, nc_att_t>
get_atts( std::string
id =
"")
const
660 void del_att(std::string att_name, std::string
id =
"")
662 int varid = name2varid(
id);
663 auto name = att_name.c_str();
665 err = nc_del_att( m_grp, varid, name);
671 int varid = name2varid(
id);
673 int retval = nc_inq_attid( m_grp, varid, att_name.c_str(), &attid);
674 return retval == NC_NOERR;
679 new_att_name, std::string
id =
"")
681 int varid = name2varid(
id);
682 auto old_name = old_att_name.c_str();
683 auto new_name = new_att_name.c_str();
685 err = nc_rename_att( m_grp, varid, old_name, new_name);
701 template<
class T,
class Attributes = std::map<std::
string, nc_att_t>>
703 const std::vector<std::string>& dim_names,
704 const Attributes& atts = {})
706 def_var( name, detail::getNCDataType<T>(), dim_names, atts);
720 template<
class Attributes = std::map<std::
string, nc_att_t>>
721 void def_var( std::string name, nc_type xtype,
722 const std::vector<std::string>& dim_names,
723 const Attributes& atts = {})
726 std::vector<int> dimids( dim_names.size());
727 for(
unsigned u=0; u<dim_names.size(); u++)
728 err = nc_inq_dimid( m_grp, dim_names[u].c_str(), &dimids[u]);
730 err = nc_def_var( m_grp, name.c_str(), xtype, dim_names.size(),
746 template<
class ContainerType, std::enable_if_t< dg::is_vector_v<
747 ContainerType, SharedVectorTag>,
bool > =
true>
749 const ContainerType& data)
751 int varid = name2varid( name);
754 err = nc_inq_varndims( m_grp, varid, &ndims);
755 assert( (
unsigned)ndims == slab.
ndim());
756 if constexpr ( dg::has_policy_v<ContainerType, dg::CudaTag>)
758 using value_type = dg::get_value_type<ContainerType>;
759 m_buffer.template set<value_type>( data.size());
760 auto& buffer = m_buffer.template get<value_type>( );
761 dg::assign ( data, buffer);
762 err = detail::put_vara_T( m_grp, varid, slab.
startp(), slab.
countp(),
766 err = detail::put_vara_T( m_grp, varid, slab.
startp(), slab.
countp(),
767 thrust::raw_pointer_cast(data.data()));
781 template<
class ContainerType,
class Attributes = std::map<std::
string, nc_att_t>,
782 std::enable_if_t< dg::is_vector_v<ContainerType, SharedVectorTag>,
bool > = true>
783 void defput_var( std::string name,
const std::vector<std::string>& dim_names,
785 const ContainerType& data)
802 template<
class T, std::enable_if_t< dg::is_scalar_v<T>,
bool> = true>
803 void put_var( std::string name,
const std::vector<size_t>& start, T data)
805 int varid = name2varid( name);
807 std::vector<size_t> count( start.size(), 1);
808 err = detail::put_vara_T( m_grp, varid, &start[0], &count[0], &data);
829 template<
class T,
class Attributes = std::map<std::
string, nc_att_t>>
852 template<
class ContainerType,
class Attributes = std::map<std::
string, nc_att_t>>
854 const Attributes& atts,
855 const ContainerType& abscissas)
858 abscissas.size(), atts);
859 put_var( name, {abscissas}, abscissas);
871 template<
class ContainerType, std::enable_if_t< dg::is_vector_v<
872 ContainerType, SharedVectorTag>,
bool > =
true>
874 ContainerType& data)
const
876 int varid = name2varid( name);
879 err = nc_inq_varndims( m_grp, varid, &ndims);
880 assert( (
unsigned)ndims == slab.
ndim());
882 for(
unsigned u=0; u<slab.
ndim(); u++)
883 size *= slab.
count()[u];
884 if constexpr ( dg::has_policy_v<ContainerType, dg::CudaTag>)
886 using value_type = dg::get_value_type<ContainerType>;
887 m_buffer.template set<value_type>( size);
888 auto& buffer = m_buffer.template get<value_type>( );
889 err = detail::get_vara_T( m_grp, varid, slab.
startp(), slab.
countp(),
891 dg::assign ( buffer, data);
896 err = detail::get_vara_T( m_grp, varid, slab.
startp(), slab.
countp(),
897 thrust::raw_pointer_cast(data.data()));
909 template<
class T, std::enable_if_t< dg::is_scalar_v<T>,
bool> = true>
910 void get_var( std::string name,
const std::vector<size_t>& start, T& data)
const
912 int varid = name2varid( name);
915 err = nc_inq_varndims( m_grp, varid, &ndims);
916 assert( (
unsigned)ndims == start.size());
918 err = detail::get_vara_T( m_grp, varid, NULL, NULL, &data);
921 std::vector<size_t> count( start.size(), 1);
922 err = detail::get_vara_T( m_grp, varid, &start[0], &count[0], &data);
932 int retval = nc_inq_varid( m_grp, name.c_str(), &varid);
933 return retval == NC_NOERR;
940 int varid = name2varid( name);
943 err = nc_inq_vartype( m_grp, varid, &xtype);
955 int varid = name2varid( name);
959 err = nc_inq_varndims( m_grp, varid, &ndims);
963 err = nc_inq_vardimid( m_grp, varid, dimids);
965 std::vector<std::string> dims(ndims);
966 for(
int i=0; i<ndims; i++)
968 char dimname [ NC_MAX_NAME+1];
969 err = nc_inq_dimname( m_grp, dimids[i], dimname);
987 return get_var_names_private( m_grp);
991 int name2varid( std::string
id)
const
1003 err = nc_inq_varid( m_grp,
id.c_str(), &varid);
1009 std::filesystem::path get_grp_path(
int ncid )
const
1012 NC_Error_Handle err;
1013 err = nc_inq_grpname_full( ncid, &len, NULL);
1014 std::string current( len,
'x');
1015 err = nc_inq_grpname_full( ncid, &len, ¤t[0]);
1019 std::map<int, std::filesystem::path> get_grps_abs(
int ncid )
const
1021 NC_Error_Handle err;
1023 err = nc_inq_grps( ncid, &num_grps, NULL);
1026 std::vector<int> group_ids( num_grps);
1027 err = nc_inq_grps( ncid, &num_grps, &group_ids[0]);
1028 std::map<int, std::filesystem::path> groups;
1029 for(
int i=0; i<num_grps; i++)
1032 err = nc_inq_grpname_full( group_ids[i], &len, NULL);
1033 std::string name( len,
'z');
1034 err = nc_inq_grpname_full( group_ids[i], &len, &name[0]);
1035 groups[group_ids[i]] = name;
1039 std::map<int, std::filesystem::path> get_grps_abs_r(
int ncid)
const
1041 auto grps = get_grps_abs(ncid);
1042 for(
auto grp : grps)
1044 auto subgrps = get_grps_abs_r( grp.first);
1045 grps.merge( subgrps);
1050 std::list<std::string> get_var_names_private(
int grpid)
const
1052 std::list<std::string> vars;
1053 int num_vars = 0, num_dims;
1054 file::NC_Error_Handle err;
1055 err = nc_inq(grpid, &num_dims, &num_vars, NULL, NULL);
1057 for(
int i=0; i<num_vars; i++)
1059 char name[NC_MAX_NAME+1];
1060 err = nc_inq_varname( grpid, i, name);
1061 vars.push_back( name);
1067 void check_open()
const
1070 throw NC_Error( 1000);
1073 bool m_open =
false;
1081 mutable dg::detail::AnyVector<thrust::host_vector> m_buffer;
1094using NcFile = SerialNcFile;
MPINcFile NcFile
Definition nc_mpi_file.h:686
NcFileMode
NetCDF file format.
Definition nc_file.h:89
@ nc_clobber
NC_CLOBBER Create a new file for read and write access, overwrite if file exists.
Definition nc_file.h:92
@ nc_nowrite
NC_NOWRITE Open an existing file for read-only access, fail if it does not exist.
Definition nc_file.h:90
@ nc_write
NC_WRITE Open an existing file for read and write access, fail if it does not exist.
Definition nc_file.h:91
@ nc_noclobber
NC_NOCLOBBER Create new file for read and write access, fail if already exists.
Definition nc_file.h:93
Definition easy_atts.h:15
DEPRECATED Empty utitlity class that handles return values of netcdf functions and throws NC_Error(st...
Definition nc_error.h:70
Class thrown by the NC_Error_Handle.
Definition nc_error.h:20
char const * what() const noexcept
What string.
Definition nc_error.h:35
A NetCDF Hyperslab for SerialNcFile.
Definition nc_hyperslab.h:27
const size_t * countp() const
Definition nc_hyperslab.h:134
const std::vector< size_t > & count() const
Definition nc_hyperslab.h:126
const size_t * startp() const
Definition nc_hyperslab.h:132
unsigned ndim() const
Definition nc_hyperslab.h:121
Serial NetCDF-4 file.
Definition nc_file.h:117
T get_att_as(std::string att_name, std::string id="") const
Get an attribute named att_name of the group or variable id.
Definition nc_file.h:611
void sync()
Call nc_sync.
Definition nc_file.h:247
void get_var(std::string name, const std::vector< size_t > &start, T &data) const
Read scalar from position start from variable named name.
Definition nc_file.h:910
std::vector< std::string > get_var_dims(std::string name) const
Get the dimension names associated to variable name.
Definition nc_file.h:953
void def_grp(std::string name)
Definition nc_file.h:270
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.
Definition nc_file.h:783
void put_att(const std::tuple< S, nc_type, T > &att, std::string id="")
Put an individual attribute of preset type to variable id.
Definition nc_file.h:571
void put_att(const std::pair< std::string, nc_att_t > &att, std::string id="")
Put an individual attribute.
Definition nc_file.h:555
void del_att(std::string att_name, std::string id="")
Remove an attribute named att_name from variable id.
Definition nc_file.h:660
SerialNcFile(SerialNcFile &&rhs)=default
Swap resources between two file handles.
std::vector< size_t > get_dims_shape(const std::vector< std::string > &dims) const
Get the size of each dimension in dims.
Definition nc_file.h:466
void rename_dim(std::string old_name, std::string new_name)
Rename a dimension from old_name to new_name.
Definition nc_file.h:438
void rename_att(std::string old_att_name, std::string new_att_name, std::string id="")
Definition nc_file.h:678
std::list< std::filesystem::path > get_grps_r() const
Get all subgroups recursively in the current group as absolute paths.
Definition nc_file.h:407
SerialNcFile(const SerialNcFile &rhs)=delete
There can only be exactly one file handle per physical file.
bool var_is_defined(std::string name) const
Definition nc_file.h:928
std::list< std::string > get_var_names() const
Get a list of variable names in the current group.
Definition nc_file.h:984
void put_atts(const Attributes &atts, std::string id="")
Write a collection of attributes to a NetCDF variable or file.
Definition nc_file.h:591
nc_type get_var_type(std::string name) const
Definition nc_file.h:938
std::map< std::string, nc_att_t > get_atts(std::string id="") const
Definition nc_file.h:645
bool is_open() const noexcept
Definition nc_file.h:218
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.
Definition nc_file.h:721
void def_grp_p(std::filesystem::path path)
Definition nc_file.h:289
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.
Definition nc_file.h:853
void set_grp(std::filesystem::path path="")
Change group to path.
Definition nc_file.h:337
std::filesystem::path get_current_path() const
Get the absolute path of the current group.
Definition nc_file.h:379
bool dim_is_defined(std::string name) const
Definition nc_file.h:541
void def_dim(std::string name, size_t size)
Define a dimension named name of size size.
Definition nc_file.h:430
void get_var(std::string name, const NcHyperslab &slab, ContainerType &data) const
Read hyperslab slab from variable named name into container data.
Definition nc_file.h:873
~SerialNcFile()
Close open nc file and release all resources.
Definition nc_file.h:159
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);
Definition nc_file.h:619
int get_grpid() const noexcept
Get the NetCDF-C ID of the current group.
Definition nc_file.h:376
SerialNcFile(const std::filesystem::path &filename, enum NcFileMode mode=nc_nowrite)
Open/Create a netCDF file.
Definition nc_file.h:135
void put_var(std::string name, const std::vector< size_t > &start, T data)
Write a single data point.
Definition nc_file.h:803
void close()
Explicitly close a file.
Definition nc_file.h:233
SerialNcFile & operator=(const SerialNcFile &rhs)=delete
There can only be exactly one file handle per physical file.
bool grp_is_defined(std::filesystem::path path) const
Check for existence of the group given by path.
Definition nc_file.h:315
bool att_is_defined(std::string att_name, std::string id="") const
Definition nc_file.h:669
std::vector< std::string > get_unlim_dims() const
Definition nc_file.h:515
void put_var(std::string name, const NcHyperslab &slab, const ContainerType &data)
Write data to a variable.
Definition nc_file.h:748
std::vector< std::string > get_dims(bool include_parents=true) const
Definition nc_file.h:483
void def_dimvar_as(std::string name, size_t size, const Attributes &atts)
Define a dimension and dimension variable in one go.
Definition nc_file.h:830
int get_ncid() const noexcept
Get the ncid of the underlying NetCDF C-API.
Definition nc_file.h:259
size_t get_dim_size(std::string name) const
Get the size of the dimension named name.
Definition nc_file.h:453
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.
Definition nc_file.h:702
std::map< std::string, T > get_atts_as(std::string id="") const
Read all NetCDF attributes of a certain type.
Definition nc_file.h:637
std::list< std::filesystem::path > get_grps() const
Definition nc_file.h:387
void rename_grp(std::string old_name, std::string new_name)
Definition nc_file.h:366
void open(const std::filesystem::path &filename, enum NcFileMode mode=nc_nowrite)
Definition nc_file.h:185