Extension: Json and NetCDF utilities
#include "dg/file/file.h"
Loading...
Searching...
No Matches
nc_file.h
Go to the documentation of this file.
1#pragma once
2
3#include <filesystem>
4#include <functional>
5#include <list>
6#include "../dg/blas.h"
7#include "../dg/backend/memory.h"
8#include "nc_error.h"
9#include "easy_atts.h"
10#include "easy_output.h"
11#include "easy_input.h"
12
19namespace dg
20{
21namespace file
22{
95
117{
119 // ///////////////////////////// CONSTRUCTORS/DESTRUCTOR /////////
122 SerialNcFile () = default;
135 SerialNcFile(const std::filesystem::path& filename,
136 enum NcFileMode mode = nc_nowrite)
137 {
138 open( filename, mode);
139 }
146 SerialNcFile(const SerialNcFile& rhs) = delete;
148 SerialNcFile& operator =(const SerialNcFile & rhs) = delete;
149
152 SerialNcFile(SerialNcFile&& rhs) = default;
155
160 {
161 try
162 {
163 close();
164 }
165 catch (NC_Error &e)
166 {
167 std::cerr << e.what() << std::endl;
168 }
169 }
170 // /////////////////// open/close /////////
171
185 void open(const std::filesystem::path& filename,
186 enum NcFileMode mode = nc_nowrite)
187 {
188 // Like a std::fstream opening fails if file already associated
189 if( m_open)
190 throw NC_Error( 1002);
191
192 // TODO Test the pathing on Windows
193 NC_Error_Handle err;
194 switch (mode)
195 {
196 case nc_nowrite:
197 err = nc_open( filename.string().c_str(), NC_NETCDF4 |
198 NC_NOWRITE, &m_ncid);
199 break;
200 case nc_write:
201 err = nc_open( filename.string().c_str(), NC_NETCDF4 |
202 NC_WRITE, &m_ncid);
203 break;
204 case nc_noclobber:
205 err = nc_create( filename.string().c_str(), NC_NETCDF4 |
206 NC_NOCLOBBER, &m_ncid);
207 break;
208 case nc_clobber:
209 err = nc_create( filename.string().c_str(), NC_NETCDF4 |
210 NC_CLOBBER, &m_ncid);
211 break;
212 }
213 m_open = true;
214 m_grp = m_ncid;
215 }
218 bool is_open() const noexcept
219 {
220 // is_closed() == not is_open()
221 return m_open;
222 }
223
233 void close()
234 {
235 // when the file is closed the ncid may be assigned to a new file
236 if (m_open)
237 {
238 NC_Error_Handle err;
239 err = nc_close(m_ncid);
240 }
241
242 m_open = false;
243 m_grp = m_ncid = 0;
244 }
245
247 void sync()
248 {
249 check_open();
250 NC_Error_Handle err;
251 err = nc_sync( m_ncid);
252 }
259 int get_ncid() const noexcept{ return m_ncid;}
260
261 // ///////////// Groups /////////////////
270 void def_grp( std::string name)
271 {
272 check_open();
273 int new_grp_id = 0;
274 NC_Error_Handle err;
275 err = nc_def_grp( m_grp, name.c_str(), &new_grp_id);
276 // we just forget the id => always need to ask netcdf for id
277 // Is no performance hit as cached
278 }
279
289 void def_grp_p( std::filesystem::path path)
290 {
291 check_open();
292 if( not path.has_root_path()) // it is a relative path
293 {
294 auto current = get_current_path();
295 path = (current / path);
296 }
297 int groupid = m_ncid;
298 auto rel_path = path.relative_path();
299 NC_Error_Handle err;
300 for( auto it = rel_path.begin(); it != rel_path.end(); it++)
301 {
302 std::string grp = *it;
303 int new_grpid;
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);
307 else
308 groupid = new_grpid;
309 }
310 }
315 bool grp_is_defined( std::filesystem::path path) const
316 {
317 check_open();
318 std::string name = path.generic_string();
319 if( not path.has_root_path()) // it is a relative path
320 {
321 auto current = get_current_path();
322 name = (current / path ).generic_string();
323 }
324 int grpid=0;
325 int retval = nc_inq_grp_full_ncid( m_ncid, name.c_str(), &grpid);
326 return retval == NC_NOERR;
327 }
337 void set_grp( std::filesystem::path path = "")
338 {
339 check_open();
340 NC_Error_Handle err;
341 std::string name = path.generic_string();
342 if ( name == ".")
343 return;
344 if ( name == "" or name == "/")
345 {
346 m_grp = m_ncid;
347 }
348 else if ( name == "..")
349 {
350 if( m_grp == m_ncid)
351 return;
352 err = nc_inq_grp_parent( m_grp, &m_grp);
353 }
354 else
355 {
356 if( not path.has_root_path()) // it is a relative path
357 {
358 auto current = get_current_path();
359 name = (current / path ).generic_string();
360 }
361 err = nc_inq_grp_full_ncid( m_ncid, name.c_str(), &m_grp);
362 }
363 }
366 void rename_grp( std::string old_name, std::string new_name)
367 {
368 check_open();
369 NC_Error_Handle err;
370 int old_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());
373 }
374
376 int get_grpid() const noexcept{ return m_grp;}
377
379 std::filesystem::path get_current_path( ) const
380 {
381 check_open();
382 return get_grp_path( m_grp);
383 }
384
387 std::list<std::filesystem::path> get_grps( ) const
388 {
389 check_open();
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);
394 return grps_v;
395
396 }
407 std::list<std::filesystem::path> get_grps_r( ) const
408 {
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);
413 return grps_v;
414 }
415
416 // //////////// Dimensions ////////////////////////
430 void def_dim( std::string name, size_t size)
431 {
432 check_open();
433 NC_Error_Handle err;
434 int dim_id;
435 err = nc_def_dim( m_grp, name.c_str(), size, &dim_id);
436 }
438 void rename_dim( std::string old_name, std::string new_name)
439 {
440 check_open();
441 int dimid;
442 NC_Error_Handle err;
443 err = nc_inq_dimid( m_grp, old_name.c_str(), &dimid);
444 err = nc_rename_dim( m_grp, dimid, new_name.c_str());
445 }
453 size_t get_dim_size( std::string name) const
454 {
455 check_open();
456 NC_Error_Handle err;
457 int dimid;
458 err = nc_inq_dimid( m_grp, name.c_str(), &dimid);
459 size_t len;
460 err = nc_inq_dimlen( m_grp, dimid, &len);
461 return len;
462
463 }
464
466 std::vector<size_t> get_dims_shape( const std::vector<std::string>& dims) const
467 {
468 std::vector<size_t> shape( dims.size());
469 for( unsigned u=0; u<dims.size(); u++)
470 shape[u] = get_dim_size( dims[u]);
471 return shape;
472 }
483 std::vector<std::string> get_dims( bool include_parents = true) const
484 {
485 check_open();
486 NC_Error_Handle err;
487 // There is a correct way of getting the dimensions in a group
488 // https://github.com/Unidata/netcdf-c/issues/2873
489 int ndims;
490 err = nc_inq_dimids( m_grp, &ndims, NULL, include_parents);
491 if( ndims == 0)
492 return {};
493 int dimids[ndims];
494 err = nc_inq_dimids( m_grp, &ndims, dimids, include_parents);
495 // Globally dimension ids are 0, 1, 2, ... in the order in which the
496 // dimensions were defined
497 std::vector<std::string> dims;
498 for ( int i = 0; i < ndims; i++)
499 {
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) ) ==
503 dims.end())
504 dims.push_back(dimname);
505 }
506 return dims;
507 }
508
515 std::vector<std::string> get_unlim_dims( ) const
516 {
517 check_open();
518 int ndims;
519 NC_Error_Handle err;
520 err = nc_inq_unlimdims( m_grp, &ndims, NULL);
521 if( ndims == 0)
522 return {};
523 int dimids[ndims];
524 // Our tests indicate that this does not return the unlimited dimensions
525 // of the parent group even though the documentation says so...
526 err = nc_inq_unlimdims( m_grp, &ndims, dimids);
527 std::vector<std::string> dims;
528 for( int i=0; i<ndims; i++)
529 {
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) ) ==
533 dims.end())
534 dims.push_back(dimname);
535 }
536 return dims;
537 }
538
541 bool dim_is_defined( std::string name) const
542 {
543 int dimid=0;
544 int retval = nc_inq_dimid( m_grp, name.c_str(), &dimid);
545 return retval == NC_NOERR;
546 }
548
555 void put_att( const std::pair<std::string, nc_att_t>& att, std::string id = "")
556 {
557 int varid = name2varid( id);
558 dg::file::detail::put_att( m_grp, varid, att);
559 }
560
570 template<class S, class T> // T cannot be nc_att_t
571 void put_att( const std::tuple<S,nc_type, T>& att, std::string id = "")
572 {
573 int varid = name2varid( id);
574 dg::file::detail::put_att( m_grp, varid, att);
575 }
590 template<class Attributes = std::map<std::string, nc_att_t> > // *it must be usable in put_att
591 void put_atts(const Attributes& atts, std::string id = "")
592 {
593 int varid = name2varid( id);
594 dg::file::detail::put_atts( m_grp, varid, atts);
595 }
596
597 // ///////////////// Attribute getters
598
610 template<class T>
611 T get_att_as(std::string att_name, std::string id = "") const
612 {
613 int varid = name2varid( id);
614 return dg::file::detail::get_att_as<T>( m_grp, varid, att_name);
615 }
616
618 template<class T>
619 std::vector<T> get_att_vec_as(std::string att_name, std::string id = "") const
620 {
621 return get_att_as<std::vector<T>>( att_name, id);
622 }
623
636 template<class T>
637 std::map<std::string, T> get_atts_as( std::string id = "") const
638 {
639 int varid = name2varid( id);
640 return dg::file::detail::get_atts_as<T>( m_grp, varid);
641 }
642
645 std::map<std::string, nc_att_t> get_atts( std::string id = "") const
646 {
647 return get_atts_as<nc_att_t>( id);
648 }
649
660 void del_att(std::string att_name, std::string id = "")
661 {
662 int varid = name2varid( id);
663 auto name = att_name.c_str();
664 NC_Error_Handle err;
665 err = nc_del_att( m_grp, varid, name);
666 }
669 bool att_is_defined(std::string att_name, std::string id = "") const
670 {
671 int varid = name2varid( id);
672 int attid;
673 int retval = nc_inq_attid( m_grp, varid, att_name.c_str(), &attid);
674 return retval == NC_NOERR;
675 }
678 void rename_att(std::string old_att_name, std::string
679 new_att_name, std::string id = "")
680 {
681 int varid = name2varid( id);
682 auto old_name = old_att_name.c_str();
683 auto new_name = new_att_name.c_str();
684 NC_Error_Handle err;
685 err = nc_rename_att( m_grp, varid, old_name, new_name);
686 }
687
688 // //////////// Variables ////////////////////////
701 template<class T, class Attributes = std::map<std::string, nc_att_t>>
702 void def_var_as( std::string name,
703 const std::vector<std::string>& dim_names,
704 const Attributes& atts = {})
705 {
706 def_var( name, detail::getNCDataType<T>(), dim_names, atts);
707 }
708
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 = {})
724 {
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]);
729 int varid;
730 err = nc_def_var( m_grp, name.c_str(), xtype, dim_names.size(),
731 &dimids[0], &varid);
732 put_atts( atts, name);
733 }
734
746 template<class ContainerType, std::enable_if_t< dg::is_vector_v<
747 ContainerType, SharedVectorTag>, bool > = true>
748 void put_var( std::string name, const NcHyperslab& slab,
749 const ContainerType& data)
750 {
751 int varid = name2varid( name);
753 int ndims;
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>)
757 {
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(),
763 buffer.data());
764 }
765 else
766 err = detail::put_vara_T( m_grp, varid, slab.startp(), slab.countp(),
767 thrust::raw_pointer_cast(data.data()));
768 }
769
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,
784 const Attributes& atts, const NcHyperslab& slab,
785 const ContainerType& data)
786 {
787 def_var_as<dg::get_value_type<ContainerType>>( name, dim_names, atts);
788 put_var( name, slab, data);
789 }
790
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)
804 {
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);
809 }
810
829 template<class T, class Attributes = std::map<std::string, nc_att_t>>
830 void def_dimvar_as( std::string name, size_t size, const Attributes& atts)
831 {
832 def_dim( name, size);
833 def_var_as<T>( name, {name}, atts);
834 }
835
852 template<class ContainerType, class Attributes = std::map<std::string, nc_att_t>>
853 void defput_dim( std::string name,
854 const Attributes& atts,
855 const ContainerType& abscissas)
856 {
858 abscissas.size(), atts);
859 put_var( name, {abscissas}, abscissas);
860 }
861
871 template<class ContainerType, std::enable_if_t< dg::is_vector_v<
872 ContainerType, SharedVectorTag>, bool > = true>
873 void get_var( std::string name, const NcHyperslab& slab,
874 ContainerType& data) const
875 {
876 int varid = name2varid( name);
878 int ndims;
879 err = nc_inq_varndims( m_grp, varid, &ndims);
880 assert( (unsigned)ndims == slab.ndim());
881 unsigned size = 1;
882 for( unsigned u=0; u<slab.ndim(); u++)
883 size *= slab.count()[u];
884 if constexpr ( dg::has_policy_v<ContainerType, dg::CudaTag>)
885 {
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(),
890 buffer.data());
891 dg::assign ( buffer, data);
892 }
893 else
894 {
895 data.resize( size);
896 err = detail::get_vara_T( m_grp, varid, slab.startp(), slab.countp(),
897 thrust::raw_pointer_cast(data.data()));
898 }
899 }
900
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
911 {
912 int varid = name2varid( name);
914 int ndims;
915 err = nc_inq_varndims( m_grp, varid, &ndims);
916 assert( (unsigned)ndims == start.size());
917 if( ndims == 0)
918 err = detail::get_vara_T( m_grp, varid, NULL, NULL, &data);
919 else
920 {
921 std::vector<size_t> count( start.size(), 1);
922 err = detail::get_vara_T( m_grp, varid, &start[0], &count[0], &data);
923 }
924 }
925
928 bool var_is_defined( std::string name) const
929 {
930 check_open();
931 int varid=0;
932 int retval = nc_inq_varid( m_grp, name.c_str(), &varid);
933 return retval == NC_NOERR;
934 }
935
938 nc_type get_var_type(std::string name) const
939 {
940 int varid = name2varid( name);
941 int xtype;
943 err = nc_inq_vartype( m_grp, varid, &xtype);
944 return xtype;
945 }
953 std::vector<std::string> get_var_dims(std::string name) const
954 {
955 int varid = name2varid( name);
957
958 int ndims;
959 err = nc_inq_varndims( m_grp, varid, &ndims);
960 if( ndims == 0)
961 return {};
962 int dimids[ndims];
963 err = nc_inq_vardimid( m_grp, varid, dimids);
964
965 std::vector<std::string> dims(ndims);
966 for( int i=0; i<ndims; i++)
967 {
968 char dimname [ NC_MAX_NAME+1];
969 err = nc_inq_dimname( m_grp, dimids[i], dimname);
970 dims[i] = dimname;
971 }
972 return dims;
973 }
974
984 std::list<std::string> get_var_names() const
985 {
986 check_open();
987 return get_var_names_private( m_grp);
988 }
989
990 private:
991 int name2varid( std::string id) const
992 {
993 // Variable ids are persistent once created
994 // Attribute ids can change if one deletes attributes
995 // This is fast even for lots variables (1000 vars take <ms)
996 check_open();
997 NC_Error_Handle err;
998 int varid;
999 if ( id == "")
1000 varid = NC_GLOBAL;
1001 else
1002 {
1003 err = nc_inq_varid( m_grp, id.c_str(), &varid);
1004 }
1005 return varid;
1006
1007 }
1008 // Absolute current path
1009 std::filesystem::path get_grp_path( int ncid ) const
1010 {
1011 size_t len;
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, &current[0]);
1016 return current;
1017 }
1018 // Absolute paths of subgroups
1019 std::map<int, std::filesystem::path> get_grps_abs( int ncid ) const
1020 {
1021 NC_Error_Handle err;
1022 int num_grps;
1023 err = nc_inq_grps( ncid, &num_grps, NULL);
1024 if( num_grps == 0)
1025 return {};
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++)
1030 {
1031 size_t len;
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;
1036 }
1037 return groups;
1038 }
1039 std::map<int, std::filesystem::path> get_grps_abs_r( int ncid) const
1040 {
1041 auto grps = get_grps_abs(ncid);
1042 for( auto grp : grps)
1043 {
1044 auto subgrps = get_grps_abs_r( grp.first);
1045 grps.merge( subgrps);
1046 }
1047 return grps;
1048 }
1049 // Need a different name so mpi_file can invoke
1050 std::list<std::string> get_var_names_private(int grpid) const
1051 {
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);
1056 // https://docs.unidata.ucar.edu/netcdf-c/current/reading_unknown.html
1057 for( int i=0; i<num_vars; i++)
1058 {
1059 char name[NC_MAX_NAME+1]; // 256
1060 err = nc_inq_varname( grpid, i, name);
1061 vars.push_back( name);
1062 }
1063 return vars;
1064 }
1065
1066
1067 void check_open() const
1068 {
1069 if( !m_open)
1070 throw NC_Error( 1000);
1071 }
1072
1073 bool m_open = false;
1074 int m_ncid = 0; // ncid can change by closing and re-opening a file
1075 int m_grp = 0;
1076 // the currently active group (All group ids in all open files are unique
1077 // and thus group ids can be different by opening the same file twice),
1078 // dims can be seen by all child groups
1079
1080 // Buffer for device to host transfer, and dg::assign
1081 mutable dg::detail::AnyVector<thrust::host_vector> m_buffer;
1082 // ABANDONED: Variable tracker (persists on closing and opening a different
1083 // file) The problem with trying to track is
1084 // 1) do we track every file that is opened?
1085 // 2) we cannot prevent someone from opening closing a file here and simply
1086 // opening the same file somewhere else. The info is lost or corrupted then
1087 //std::map<std::filesystem::path, std::map<std::filesystem::path, NcVarInfo>>
1088 // m_vars;
1089};
1090
1091#ifndef MPI_VERSION
1094using NcFile = SerialNcFile;
1095#endif // MPI_VERSION
1096
1097}// namespace file
1098}// namespace dg
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