-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathncdds.cc
661 lines (570 loc) · 23.7 KB
/
ncdds.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
// -*- mode: c++; c-basic-offset:4 -*-
// This file is part of nc_handler, a data handler for the OPeNDAP data
// server.
// Copyright (c) 2002,2003 OPeNDAP, Inc.
// Author: James Gallagher <jgallagher@opendap.org>
//
// This is free software; you can redistribute it and/or modify it under the
// terms of the GNU Lesser General Public License as published by the Free
// Software Foundation; either version 2.1 of the License, or (at your
// option) any later version.
//
// This software is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
// or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
// License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
//
// You can contact OPeNDAP, Inc. at PO Box 112, Saunderstown, RI. 02874-0112.
// (c) COPYRIGHT URI/MIT 1994-1996
// Please read the full copyright statement in the file COPYRIGHT.
//
// Authors:
// reza Reza Nekovei (reza@intcomm.net)
// This file contains functions which read the variables and their description
// from a netcdf file and build the in-memory DDS. These functions form the
// core of the server-side software necessary to extract the DDS from a
// netcdf data file.
//
// It also contains test code which will print the in-memory DDS to
// stdout.
//
// ReZa 10/20/94
#include "config_nc.h"
#include <cstdio>
#include <cstring>
#include <iostream>
#include <string>
#include <algorithm>
#include <netcdf.h>
#include <DDS.h>
#include <mime_util.h>
#include <util.h>
#include <BESDebug.h>
#include "NCRequestHandler.h"
#include "nc_util.h"
#include "NCInt32.h"
#include "NCUInt32.h"
#include "NCInt16.h"
#include "NCUInt16.h"
#include "NCFloat64.h"
#include "NCFloat32.h"
#include "NCByte.h"
#include "NCArray.h"
#include "NCGrid.h"
#include "NCStr.h"
#include "NCStructure.h"
using namespace libdap ;
/** This function returns the appropriate DODS BaseType for the given
netCDF data type. */
static BaseType *
build_scalar(const string &varname, const string &dataset, nc_type datatype)
{
switch (datatype) {
#if NETCDF_VERSION >= 4
case NC_STRING:
#endif
case NC_CHAR:
return (new NCStr(varname, dataset));
case NC_BYTE:
if (NCRequestHandler::get_promote_byte_to_short()) {
return (new NCInt16(varname, dataset));
}
else {
return (new NCByte(varname, dataset));
}
case NC_SHORT:
return (new NCInt16(varname, dataset));
case NC_INT:
return (new NCInt32(varname, dataset));
#if NETCDF_VERSION >= 4
case NC_UBYTE:
// NB: the dods_byte type is unsigned
return (new NCByte(varname, dataset));
case NC_USHORT:
return (new NCUInt16(varname, dataset));
case NC_UINT:
return (new NCUInt32(varname, dataset));
#endif
case NC_FLOAT:
return (new NCFloat32(varname, dataset));
case NC_DOUBLE:
return (new NCFloat64(varname, dataset));
#if NETCDF_VERSION >= 4
case NC_INT64:
case NC_UINT64:
if (NCRequestHandler::get_ignore_unknown_types())
cerr << "The netCDF handler does not currently support 64 bit integers.";
else
throw Error("The netCDF handler does not currently support 64 bit integers.");
break;
#endif
default:
throw InternalErr(__FILE__, __LINE__, "Unknown type (" + long_to_string(datatype) + ") for variable '" + varname + "'");
}
return 0;
}
#if 0
// Replaced by code in nc_util.cc. jhrg 2/9/12
static bool is_user_defined(nc_type type)
{
#if NETCDF_VERSION >= 4
return type >= NC_FIRSTUSERTYPEID;
#else
return false;
#endif
}
#endif
/** Build a grid given that one has been found. The Grid's Array is already
allocated and is passed in along with a number of arrays containing
information about the dimensions of the Grid.
Note: The dim_szs and dim_nms arrays could be removed since that information
is already in the Array ar. */
static Grid *build_grid(Array *ar, int ndims, const nc_type array_type,
const char map_names[MAX_NC_VARS][MAX_NC_NAME],
const nc_type map_types[MAX_NC_VARS],
const size_t map_sizes[MAX_VAR_DIMS],
vector<string> *all_maps)
{
// Grids of NC_CHARs are treated as Grids of strings; the outermost
// dimension (the char vector) becomes the string.
if (array_type == NC_CHAR)
--ndims;
for (int d = 0; d < ndims; ++d) {
ar->append_dim(map_sizes[d], map_names[d]);
// Save the map names for latter use, which might not happen...
all_maps->push_back(string(map_names[d]));
}
const string &filename = ar->dataset();
Grid *gr = new NCGrid(ar->name(), filename);
gr->add_var(ar, libdap::array);
// Build and add BaseType/Array instances for the maps
for (int d = 0; d < ndims; ++d) {
BaseType *local_bt = build_scalar(map_names[d], filename, map_types[d]);
NCArray *local_ar = new NCArray(local_bt->name(), filename, local_bt);
delete local_bt;
local_ar->append_dim(map_sizes[d], map_names[d]);
gr->add_var(local_ar, maps);
delete local_ar;
}
return gr;
}
#if NETCDF_VERSION >= 4
/** Build an instance of a user defined type. These can be recursively
* defined.
*/
static BaseType *build_user_defined(int ncid, int varid, nc_type xtype, const string &dataset,
int ndims, int dim_ids[MAX_VAR_DIMS])
{
size_t size;
nc_type base_type;
size_t nfields;
int class_type;
int status = nc_inq_user_type(ncid, xtype, 0/*name*/, &size, &base_type, &nfields, &class_type);
if (status != NC_NOERR)
throw InternalErr(__FILE__, __LINE__, "Could not get information about a user-defined type (" + long_to_string(status) + ").");
switch (class_type) {
case NC_COMPOUND: {
char var_name[NC_MAX_NAME+1];
nc_inq_varname(ncid, varid, var_name);
NCStructure *ncs = new NCStructure(var_name, dataset);
for (size_t i = 0; i < nfields; ++i) {
char field_name[NC_MAX_NAME+1];
nc_type field_typeid;
int field_ndims;
int field_sizes[MAX_NC_DIMS];
nc_inq_compound_field(ncid, xtype, i, field_name, 0, &field_typeid, &field_ndims, &field_sizes[0]);
BaseType *field;
if (is_user_defined_type(ncid, field_typeid)) {
//is_user_defined(field_typeid)) {
// Odd: 'varid' here seems wrong, but works.
field = build_user_defined(ncid, varid, field_typeid, dataset, field_ndims, field_sizes);
// Child compound types become anonymous variables but DAP
// requires names, so use the type name.
char var_name[NC_MAX_NAME+1];
nc_inq_compound_name(ncid, field_typeid, var_name);
field->set_name(var_name);
}
else {
field = build_scalar(field_name, dataset, field_typeid);
}
// is this a scalar or an array? Note that an array of CHAR is
// a scalar string in netcdf3.
if (field_ndims == 0 || (field_ndims == 1 && field_typeid == NC_CHAR)) {
ncs->add_var(field);
}
else {
NCArray *ar = new NCArray(field_name, dataset, field);
for (int i = 0; i < field_ndims; ++i) {
ar->append_dim(field_sizes[i]);
}
ncs->add_var(ar);
}
}
// Is this an array of a compound (DAP Structure)?
if (ndims > 0) {
NCArray *ar = new NCArray(var_name, dataset, ncs);
for (int i = 0; i < ndims; ++i) {
char dimname[NC_MAX_NAME+1];
size_t dim_sz;
int errstat = nc_inq_dim(ncid, dim_ids[i], dimname, &dim_sz);
if (errstat != NC_NOERR) {
delete ar;
throw InternalErr(__FILE__, __LINE__, string("Failed to read dimension information for the compound variable ") + var_name);
}
ar->append_dim(dim_sz, dimname);
}
return ar;
}
else {
return ncs;
}
break;
}
case NC_VLEN:
if (NCRequestHandler::get_ignore_unknown_types()) {
cerr << "in build_user_defined; found a vlen." << endl;
return 0;
}
else
throw Error("The netCDF handler does not yet suppor the NC_VLEN type.");
break;
case NC_OPAQUE: {
vector<char> name(NC_MAX_NAME+1);
status = nc_inq_varname(ncid, varid, &name[0]);
if (status != NC_NOERR)
throw InternalErr(__FILE__, __LINE__, "Could not get name of an opaque (" + long_to_string(status) + ").");
NCArray *opaque = new NCArray(&name[0], dataset, new NCByte(&name[0], dataset));
if (ndims > 0) {
for (int i = 0; i < ndims; ++i) {
char dimname[NC_MAX_NAME+1];
size_t dim_sz;
int errstat = nc_inq_dim(ncid, dim_ids[i], dimname, &dim_sz);
if (errstat != NC_NOERR) {
delete opaque;
throw InternalErr(__FILE__, __LINE__, string("Failed to read dimension information for the compound variable ") + &name[0]);
}
opaque->append_dim(dim_sz, dimname);
}
}
opaque->append_dim(size);
return opaque;
break;
}
case NC_ENUM: {
nc_type base_nc_type;
size_t base_size;
status = nc_inq_enum(ncid, xtype, 0 /*&name[0]*/, &base_nc_type, &base_size, 0/*&num_members*/);
if (status != NC_NOERR)
throw(InternalErr(__FILE__, __LINE__, "Could not get information about an enum(" + long_to_string(status) + ")."));
// get the name here - we want the var name and not the type name
vector<char> name(MAX_NC_NAME + 1);
status = nc_inq_varname(ncid, varid, &name[0]);
if (status != NC_NOERR)
throw InternalErr(__FILE__, __LINE__, "Could not get name of an opaque (" + long_to_string(status) + ").");
BaseType *enum_var = build_scalar(&name[0], dataset, base_nc_type);
if (ndims > 0) {
NCArray *ar = new NCArray(&name[0], dataset, enum_var);
for (int i = 0; i < ndims; ++i) {
char dimname[NC_MAX_NAME + 1];
size_t dim_sz;
int errstat = nc_inq_dim(ncid, dim_ids[i], dimname, &dim_sz);
if (errstat != NC_NOERR) {
delete ar;
throw InternalErr(__FILE__, __LINE__, string("Failed to read dimension information for the compound variable ") + &name[0]);
}
ar->append_dim(dim_sz, dimname);
}
return ar;
}
else {
return enum_var;
}
break;
}
default:
throw InternalErr(__FILE__, __LINE__, "Expected one of NC_COMPOUND, NC_VLEN, NC_OPAQUE or NC_ENUM");
}
return 0;
}
#endif
/** Iterate over all of the variables in the data set looking for a one
dimensional variable whose name and size match the name and size of the
dimension 'dimname' of variable 'var'. If one is found, return it's
name and size. It is considered to be a Map for 'var' (i.e., a matching
coordinate variable).
@param ncid Used with all netCDF API calls
@param var Variable number: Look for a map for this variable.
@param dimname Name of the current dimension
@param dim_sz Size of the current dimension
@param match_type Value-result parameter, holds dimension type if match found
@return true if a match coordinate variable (i.e., map) was found, false
otherwise
@note In this code I scan all the variables, maybe there's a way to look
at (only) all the shared dimensions?
*/
static bool find_matching_coordinate_variable(int ncid, int var,
char dimname[], size_t dim_sz, nc_type *match_type)
{
// For netCDF, a Grid's Map must be a netCDF dimension
int id;
// get the id matching the name.
int status = nc_inq_dimid(ncid, dimname, &id);
if (status == NC_NOERR) {
// get the length, the name was matched above
size_t length;
status = nc_inq_dimlen(ncid, id, &length);
if (status != NC_NOERR) {
string msg = "netcdf 3: could not get size for dimension ";
msg += long_to_string(id);
msg += " in variable ";
msg += string(dimname);
throw Error(msg);
}
if (length == dim_sz) {
// Both the name and size match and it's a dimension, so we've
// found our 'matching coordinate variable'. To get the type,
// Must find the variable with the name that matches the dimension.
int varid = -1;
status = nc_inq_varid(ncid, dimname, &varid);
// A variable cannot be its own coordinate variable.
// The unlimited dimension does not correspond to a variable,
// hence the status error is means the named thing is not a
// coordinate; it's not an error as far as the handler is concerned.
if (var == varid || status != NC_NOERR)
return false;
status = nc_inq_vartype(ncid, varid, match_type);
if (status != NC_NOERR) {
string msg = "netcdf 3: could not get type variable ";
msg += string(dimname);
throw Error(msg);
}
char candidate_name[MAX_NC_NAME];
nc_type candidate_nctype;
int candidate_ndims;
int candidate_dim_ids[MAX_VAR_DIMS];
int errstat = nc_inq_var(ncid, varid, candidate_name, &candidate_nctype, &candidate_ndims, candidate_dim_ids, (int *) 0);
if (errstat != NC_NOERR)
throw Error("netcdf: could not get name or dimension number for variable " + long_to_string(varid));
// Map arrays can only be one dimensional in DAP2 so if we are here
// and this thing has more than one dimension it's not a valid
// Map array for a DAP2 Grid so return false.
if(candidate_ndims!=1){
char array_name[MAX_NC_NAME];
errstat = nc_inq_varname(ncid, var, array_name);
if (errstat != NC_NOERR)
throw Error("netcdf: could not get name or dimension number for variable " + long_to_string(var));
BESDEBUG("nc", "ncdds.c - find_matching_coordinate_variable() The array '"
<< array_name << "' is not a grid because the "
"candidate Map array '" << candidate_name << " has " <<
candidate_ndims << " dimensions and DAP2 Map arrays must have "
"a single dimension" << endl);
return false;
}
return true;
}
}
return false;
}
/** Is the variable a DAP Grid?
@param ncid The open netCDF file id
@param nvars The number of variables in the file. Needed by the function
that looks for maps.
@param var The id of the variable we're asking about.
@param ndims The number of dimensions in 'var'.
@param dim_ids The dimension ids of 'var'.
@param map_sizes Value-result parameter; the size of each map.
@param map_names Value-result parameter; the name of each map.
@param map_types Value-result parameter; the type of each map.
*/
static bool is_grid(int ncid, int var, int ndims, const int dim_ids[MAX_VAR_DIMS],
size_t map_sizes[MAX_VAR_DIMS],
char map_names[MAX_NC_VARS][MAX_NC_NAME],
nc_type map_types[MAX_NC_VARS])
{
// Look at each dimension of the variable.
for (int d = 0; d < ndims; ++d) {
char dimname[MAX_NC_NAME];
size_t dim_sz;
int errstat = nc_inq_dim(ncid, dim_ids[d], dimname, &dim_sz);
if (errstat != NC_NOERR) {
string msg = "netcdf 3: could not get size for dimension ";
msg += long_to_string(d);
msg += " in variable ";
msg += long_to_string(var);
throw Error(msg);
}
nc_type match_type;
if (find_matching_coordinate_variable(ncid, var, dimname, dim_sz, &match_type)) {
map_types[d] = match_type;
map_sizes[d] = dim_sz;
strncpy(map_names[d], dimname, MAX_NC_NAME - 1);
map_names[d][MAX_NC_NAME - 1] = '\0';
}
else {
return false;
}
}
return true;
}
static bool is_dimension(const string &name, vector<string> maps)
{
vector<string>::iterator i = find(maps.begin(), maps.end(), name);
if (i != maps.end())
return true;
else
return false;
}
static NCArray *build_array(BaseType *bt, int ncid, int var,
const nc_type array_type, int ndims,
const int dim_ids[MAX_NC_DIMS])
{
NCArray *ar = new NCArray(bt->name(), bt->dataset(), bt);
if (array_type == NC_CHAR)
--ndims;
for (int d = 0; d < ndims; ++d) {
char dimname[MAX_NC_NAME];
size_t dim_sz;
int errstat = nc_inq_dim(ncid, dim_ids[d], dimname, &dim_sz);
if (errstat != NC_NOERR) {
delete ar;
throw Error("netcdf: could not get size for dimension " + long_to_string(d) + " in variable " + long_to_string(var));
}
ar->append_dim(dim_sz, dimname);
}
return ar;
}
/** Read given number of variables (nvars) from the opened netCDF file
(ncid) and add them with their appropriate type and dimensions to
the given instance of the DDS class.
@param dds_table Add variables to this DDS object
@param filename When making new variables, record this as the source
@param ncid The id of the netcdf file
@param nvars The number of variables in the opened file
*/
static void read_variables(DDS &dds_table, const string &filename, int ncid, int nvars)
{
// How this function works: The variables are scanned once but because
// netCDF includes shared dimensions as variables there are two versions
// of this function. One writes out the variables as they are found while
// the other writes scalars and Grids as they are found and saves Arrays
// for output last. Then, when writing the arrays, it checks to see if
// an array variable is also a grid dimension and, if so, does not write
// it out (thus in the second version of the function, all arrays appear
// after the other variable types and only those arrays that do not
// appear as Grid Maps are included.
// These two vectors are used to record the ids of array variables and
// the names of all of the Grid Map variables
vector<int> array_vars;
vector<string> all_maps;
// These are defined here since they are used by both loops.
char name[MAX_NC_NAME];
nc_type nctype;
int ndims;
int dim_ids[MAX_VAR_DIMS];
// Examine each variable in the file; if 'elide_grid_maps' is true, adds
// only scalars and Grids (Arrays are added in the following loop). If
// false, all variables are added in this loop.
for (int varid = 0; varid < nvars; ++varid) {
int errstat = nc_inq_var(ncid, varid, name, &nctype, &ndims, dim_ids, (int *) 0);
if (errstat != NC_NOERR)
throw Error("netcdf: could not get name or dimension number for variable " + long_to_string(varid));
// These are defined here because they are value-result parameters for
// is_grid() called below.
size_t map_sizes[MAX_VAR_DIMS];
char map_names[MAX_NC_VARS][MAX_NC_NAME];
nc_type map_types[MAX_NC_VARS];
// a scalar? NB a one-dim NC_CHAR array will have DAP type of
// dods_str_c because it's really a scalar string, not an array.
if (is_user_defined_type(ncid, nctype)) {
// is_user_defined(nctype)) {
#if NETCDF_VERSION >= 4
BaseType *bt = build_user_defined(ncid, varid, nctype, filename, ndims, dim_ids);
dds_table.add_var(bt);
delete bt;
#endif
}
else if (ndims == 0 || (ndims == 1 && nctype == NC_CHAR)) {
BaseType *bt = build_scalar(name, filename, nctype);
dds_table.add_var(bt);
delete bt;
}
else if (is_grid(ncid, varid, ndims, dim_ids, map_sizes, map_names, map_types)) {
BaseType *bt = build_scalar(name, filename, nctype);
Array *ar = new NCArray(name, filename, bt);
delete bt;
Grid *gr = build_grid(ar, ndims, nctype, map_names, map_types, map_sizes, &all_maps);
delete ar;
dds_table.add_var(gr);
delete gr;
}
else {
if (!NCRequestHandler::get_show_shared_dims()) {
array_vars.push_back(varid);
} else {
BaseType *bt = build_scalar(name, filename, nctype);
NCArray *ar = build_array(bt, ncid, varid, nctype, ndims, dim_ids);
delete bt;
dds_table.add_var(ar);
delete ar;
}
}
}
// This code is only run if elide_dimension_arrays is true and in that case the
// loop above did not create any simple arrays. Instead it pushed the
// var ids of things that look like simple arrays onto a vector. This code
// will add all of those that really are arrays and not the ones that are
// dimensions used by a Grid.
if (!NCRequestHandler::get_show_shared_dims()) {
// Now just loop through the saved array variables, writing out only
// those that are not Grid Maps
nvars = array_vars.size();
for (int i = 0; i < nvars; ++i) {
int var = array_vars.at(i);
int errstat = nc_inq_var(ncid, var, name, &nctype, &ndims,
dim_ids, (int *) 0);
if (errstat != NC_NOERR) {
string msg = "netcdf 3: could not get name or dimension number for variable ";
msg += long_to_string(var);
throw Error(msg);
}
// If an array already appears as a Grid Map, don't write it out
// as an array too.
if (is_dimension(string(name), all_maps))
continue;
BaseType *bt = build_scalar(name, filename, nctype);
Array *ar = build_array(bt, ncid, var, nctype, ndims, dim_ids);
delete bt;
dds_table.add_var(ar);
delete ar;
}
}
}
/** Given a reference to an instance of class DDS and a filename that refers
to a netcdf file, read the netcdf file and extract all the dimensions of
each of its variables. Add the variables and their dimensions to the
instance of DDS.
@param elide_dimension_arrays If true, don't include an array if it's
really a dimension used by a Grid. */
void nc_read_dataset_variables(DDS &dds_table, const string &filename)
{
ncopts = 0;
int ncid, errstat;
int nvars;
errstat = nc_open(filename.c_str(), NC_NOWRITE, &ncid);
if (errstat != NC_NOERR)
throw Error(errstat, "Could not open " + path_to_filename(filename) + ".");
// how many variables?
errstat = nc_inq_nvars(ncid, &nvars);
if (errstat != NC_NOERR)
throw Error(errstat, "Could not inquire about netcdf file: " + path_to_filename(filename) + ".");
// dataset name
dds_table.set_dataset_name(name_path(filename));
// read variables' classes
read_variables(dds_table, filename, ncid, nvars);
if (nc_close(ncid) != NC_NOERR)
throw InternalErr(__FILE__, __LINE__, "ncdds: Could not close the dataset!");
}