mirror of
https://github.com/Unidata/netcdf-c.git
synced 2025-01-06 15:34:44 +08:00
25f062528b
The file docs/indexing.dox tries to provide design information for the refactoring. The primary change is to replace all walking of linked lists with the use of the NCindex data structure. Ncindex is a combination of a hash table (for name-based lookup) and a vector (for walking the elements in the index). Additionally, global vectors are added to NC_HDF5_FILE_INFO_T to support direct mapping of an e.g. dimid to the NC_DIM_INFO_T object. These global vectors exist for dimensions, types, and groups because they have globally unique id numbers. WARNING: 1. since libsrc4 and libsrchdf4 share code, there are also changes in libsrchdf4. 2. Any outstanding pull requests that change libsrc4 or libhdf4 are likely to cause conflicts with this code. 3. The original reason for doing this was for performance improvements, but as noted elsewhere, this may not be significant because the meta-data read performance apparently is being dominated by the hdf5 library because we do bulk meta-data reading rather than lazy reading.
1394 lines
50 KiB
C
1394 lines
50 KiB
C
/*********************************************************************
|
|
* Copyright 1993, UCAR/Unidata
|
|
* See netcdf/COPYRIGHT file for copying and redistribution conditions.
|
|
* $Header: /upc/share/CVS/netcdf-3/nctst/atttests.c,v 1.18 2006/10/31 16:21:45 ed Exp $
|
|
*********************************************************************/
|
|
|
|
#ifdef _MPW
|
|
#define __SEG__ toobig /* under MPW on MacOS, makes it fit */
|
|
#endif
|
|
|
|
#include <config.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h> /* for free() */
|
|
#include "netcdf.h"
|
|
#include "testcdf.h" /* defines in-memory test cdf structure */
|
|
#include "add.h" /* functions to update in-memory netcdf */
|
|
#include "error.h"
|
|
#include "emalloc.h"
|
|
#include "tests.h"
|
|
#include "val.h"
|
|
|
|
#define LEN_OF(array) ((sizeof array) / (sizeof array[0]))
|
|
|
|
|
|
/*
|
|
* Test ncattput
|
|
* check that new attribute put works in define mode
|
|
* check that NC_GLOBAL variable id works
|
|
* check that changing type of existing attribute works in define mode
|
|
* check that increasing length of attribute works in define mode
|
|
* check that changing value of existing attribute works in define mode
|
|
* try with bad datatype, should fail
|
|
* try with negative length, should fail
|
|
* try increasing length of attribute in data mode, should fail
|
|
* try putting new attribute in data mode, should fail
|
|
* check that changing type of existing attribute works in data mode
|
|
* check that decreasing length of attribute works in data mode
|
|
* check that changing value of existing attribute works in data mode
|
|
* try with bad variable handle, should fail
|
|
* try with bad netCDF handle, check error
|
|
*/
|
|
int
|
|
test_ncattput(path)
|
|
const char *path; /* name of writable netcdf file to open */
|
|
{
|
|
int nerrs = 0;
|
|
static char pname[] = "test_ncattput";
|
|
int cdfid; /* netcdf id */
|
|
int ndims; /* number of dimensions */
|
|
int nvars; /* number of variables */
|
|
int ngatts_prev, ngatts; /* number of global attributes */
|
|
int xdimid; /* id of unlimited dimension */
|
|
int ia, id;
|
|
static char byte_vals[] = {'a', 'b'};
|
|
static char char_vals[] = "chars";
|
|
static short short_vals[] = {-999, 0, 999};
|
|
static nclong long_vals[] = {10, 20};
|
|
static float float_vals[] = {1.5, 2.5, 3.5 };
|
|
static double double_vals[] = {4.5, 5.5, 6.5, 7.5};
|
|
/*
|
|
* test attributes; it is important for this test that the size
|
|
* required for the attribute values increases monotonically.
|
|
*/
|
|
static struct cdfatt atts[] = {
|
|
{___, "att0", NC_BYTE, LEN_OF(byte_vals), (void *) byte_vals},
|
|
{___, "att1", NC_CHAR, LEN_OF(char_vals), (void *) char_vals},
|
|
{___, "att2", NC_SHORT, LEN_OF(short_vals), (void *) short_vals},
|
|
{___, "att3", NC_LONG, LEN_OF(long_vals), (void *) long_vals},
|
|
{___, "att4", NC_FLOAT, LEN_OF(float_vals), (void *) float_vals},
|
|
{___, "att5", NC_DOUBLE, LEN_OF(double_vals), (void *) double_vals}
|
|
};
|
|
int na = LEN_OF(atts); /* number of test attributes */
|
|
int ww_id; /* variable id */
|
|
static struct cdfvar ww = /* new variable */
|
|
{"ww", NC_LONG, 1, ___, 0};
|
|
static struct cdfatt tmp; /* attribute */
|
|
|
|
(void) fprintf(stderr, "*** Testing %s ...\t", &pname[5]);
|
|
|
|
if ((cdfid = ncopen(path, NC_WRITE)) == -1) {
|
|
error("%s: ncopen failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* enter define mode */
|
|
if (ncredef(cdfid) == -1) {
|
|
error("%s: cdredef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* get count of global attributes */
|
|
if (ncinquire(cdfid, &ndims, &nvars, &ngatts, &xdimid) == -1) {
|
|
error("%s: ncinquire failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
ngatts_prev = ngatts;
|
|
/* in define mode, add global attributes of every type */
|
|
for (ia = 0; ia < na; ia++) {
|
|
if (ncattput(cdfid, NC_GLOBAL, atts[ia].name, atts[ia].type,
|
|
atts[ia].len, atts[ia].val) == -1) {
|
|
error("%s: ncattput of NC_GLOBAL attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, NC_GLOBAL, &atts[ia]); /* keep in-memory netcdf updated */
|
|
}
|
|
/* make sure count of global attributes has been updated */
|
|
if (ncinquire(cdfid, &ndims, &nvars, &ngatts, &xdimid) == -1) {
|
|
error("%s: ncinquire failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ngatts != ngatts_prev + na) {
|
|
error("%s: number of global = %d, expected %d",
|
|
pname, ngatts, ngatts_prev + na);
|
|
nerrs++;
|
|
}
|
|
/* check with ncattinq and ncattget that NC_GLOBAL attributes put OK */
|
|
for (ia = 0; ia < na; ia++) {
|
|
if (ncattinq(cdfid, NC_GLOBAL, atts[ia].name,
|
|
&tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq of global attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (atts[ia].type != tmp.type || atts[ia].len != tmp.len) {
|
|
error("%s: NC_GLOBAL ncattinq got unexpected type or len",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(atts[ia].len * nctypelen(atts[ia].type));
|
|
if (ncattget(cdfid, NC_GLOBAL, atts[ia].name, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, atts[ia].val) != 0) {
|
|
error("%s: ncattget got bad values after put of NC_GLOBAL attrs",
|
|
pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
}
|
|
/* add a variable, then variable attributes of every type */
|
|
ww.dims = (int *) emalloc(sizeof(int) * ww.ndims);
|
|
for (id = 0; id < ww.ndims; id++)
|
|
ww.dims[id] = id;
|
|
if ((ww_id = ncvardef(cdfid,
|
|
ww.name, ww.type, ww.ndims, ww.dims)) == -1) {
|
|
error("%s: ncvardef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_var(&test, &ww); /* keep in-memory netcdf in sync */
|
|
for (ia = 0; ia < na; ia++) {
|
|
if (ncattput(cdfid, ww_id,
|
|
atts[ia].name, atts[ia].type, atts[ia].len, atts[ia].val)
|
|
== -1) {
|
|
error("%s: ncattput of variable attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, ww_id, &atts[ia]); /* keep in-memory netcdf updated */
|
|
}
|
|
/* check with ncattinq and ncattget that variable attributes put OK */
|
|
for (ia = 0; ia < na; ia++) {
|
|
if (ncattinq(cdfid, ww_id, atts[ia].name,
|
|
&tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq of variable attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (atts[ia].type != tmp.type || atts[ia].len != tmp.len) {
|
|
error("%s: ncattinq for new attribute got bad type or len",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(atts[ia].len * nctypelen(atts[ia].type));
|
|
if (ncattget(cdfid, ww_id, atts[ia].name, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, atts[ia].val) != 0) {
|
|
error("%s: ncattget got bad values after put of variable attrs",
|
|
pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
}
|
|
/*
|
|
* check that changing type of existing attribute, increasing
|
|
* length of attribute, and changing value of existing attribute
|
|
* work OK in define mode.
|
|
*/
|
|
tmp.name = (char *) emalloc(MAX_NC_NAME);
|
|
for (ia = 1; ia < na; ia++) {
|
|
if (ncattput(cdfid, ww_id, atts[ia-1].name, atts[ia].type,
|
|
atts[ia].len, atts[ia].val) == -1) {
|
|
error("%s: ncattput of larger attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
tmp.var = atts[ia].var;
|
|
(void) strcpy (tmp.name, atts[ia-1].name);
|
|
tmp.type = atts[ia].type;
|
|
tmp.len = atts[ia].len;
|
|
tmp.val = atts[ia].val;
|
|
add_att(&test, ww_id, &tmp); /* keep in-memory netcdf updated */
|
|
}
|
|
/* check with ncattinq and ncattget that variable attributes put OK */
|
|
for (ia = 1; ia < na; ia++) {
|
|
if (ncattinq(cdfid, ww_id, atts[ia-1].name,
|
|
&tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq of larger attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (atts[ia].type != tmp.type || atts[ia].len != tmp.len) {
|
|
error("%s: ncattinq for larger attribute got bad type or len",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(atts[ia].len * nctypelen(atts[ia].type));
|
|
if (ncattget(cdfid, ww_id, atts[ia-1].name, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, atts[ia].val) != 0) {
|
|
error("%s: ncattget got bad values after put of larger attrs",
|
|
pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
}
|
|
/* try with bad datatype, should fail */
|
|
if (ncattput(cdfid, ww_id, "bogus_att1", BAD_TYPE,
|
|
atts[0].len, atts[0].val) != -1) {
|
|
error("%s: ncattput should fail with bad type", pname);
|
|
nerrs++;
|
|
}
|
|
/* try with negative length, should fail */
|
|
if (ncattput(cdfid, ww_id, "bogus_att2", atts[0].type,
|
|
-1, atts[0].val) != -1) {
|
|
error("%s: ncattput should fail with bad length", pname);
|
|
nerrs++;
|
|
}
|
|
if (ncendef (cdfid) == -1) {
|
|
error("%s: ncendef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in data mode try increasing length of attribute, should fail */
|
|
if (ncattput(cdfid, ww_id, atts[0].name, atts[0].type,
|
|
atts[0].len + 10, atts[0].val) != -1) {
|
|
error("%s: ncattput should fail with increased length in data mode",
|
|
pname);
|
|
nerrs++;
|
|
/* reset to correct length for later tests */
|
|
if (ncattput(cdfid, ww_id, atts[0].name, atts[0].type,
|
|
atts[0].len, atts[0].val) != -1) {
|
|
error("%s: ncattput failed to reset length in data mode", pname);
|
|
nerrs++;
|
|
}
|
|
}
|
|
/* try creating new attribute in data mode, should fail */
|
|
if (ncattput(cdfid, ww_id, "new_name", atts[0].type,
|
|
atts[0].len, atts[0].val) != -1) {
|
|
error("%s: ncattput of new attribute in data mode should fail",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/*
|
|
* check that changing type of existing attribute, decreasing
|
|
* length of attribute, and changing value of existing attribute
|
|
* work OK in data mode
|
|
*/
|
|
for (ia = 0; ia < na - 1; ia++) {
|
|
if (ncattput(cdfid, ww_id, atts[ia+1].name, atts[ia].type,
|
|
atts[ia].len, atts[ia].val) == -1) {
|
|
error("%s: ncattput of smaller attribute failed in data mode",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
tmp.var = atts[ia].var;
|
|
(void) strcpy (tmp.name, atts[ia+1].name);
|
|
tmp.type = atts[ia].type;
|
|
tmp.len = atts[ia].len;
|
|
tmp.val = atts[ia].val;
|
|
add_att(&test, ww_id, &tmp); /* keep in-memory netcdf updated */
|
|
}
|
|
/* check with ncattinq and ncattget that variable attributes put OK */
|
|
for (ia = 0; ia < na - 1; ia++) {
|
|
if (ncattinq(cdfid, ww_id, atts[ia+1].name, &tmp.type, &tmp.len)
|
|
== -1) {
|
|
error("%s: ncattinq of variable attribute failed in data mode",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (atts[ia].type != tmp.type || atts[ia].len != tmp.len) {
|
|
error("%s: VARIABLE ncattinq got bad type or len in data mode",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(atts[ia].len * nctypelen(atts[ia].type));
|
|
if (ncattget(cdfid, ww_id, atts[ia+1].name, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed in data mode",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, atts[ia].val) != 0) {
|
|
error("%s: ncattget got bad values in data mode", pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
}
|
|
/* try with bad variable handle, should fail */
|
|
if (ncattput(cdfid, test.nvars, atts[0].name, atts[0].type, atts[0].len,
|
|
atts[0].val) != -1) {
|
|
error("%s: ncattput should fail with bad variable handle", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncclose (cdfid) == -1) {
|
|
error("%s: ncclose failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* try with bad netcdf handle, should fail */
|
|
if (ncattput(cdfid, ww_id, atts[0].name, atts[0].type, atts[0].len,
|
|
atts[0].val) != -1) {
|
|
error("%s: ncattput should fail with bad netcdf handle", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
free(tmp.name);
|
|
free(ww.dims);
|
|
if (nerrs > 0)
|
|
(void) fprintf(stderr,"FAILED! ***\n");
|
|
else
|
|
(void) fprintf(stderr,"ok ***\n");
|
|
|
|
return nerrs;
|
|
}
|
|
|
|
|
|
/*
|
|
* Test ncattinq
|
|
* check returned values of properly created attributes
|
|
* try with nonexisting attribute, check error
|
|
* try with bad variable handle, check error
|
|
* try with bad netCDF handle, check error
|
|
*/
|
|
int
|
|
test_ncattinq(path)
|
|
const char *path; /* name of writable netcdf file to open */
|
|
{
|
|
int nerrs = 0;
|
|
static char pname[] = "test_ncattinq";
|
|
int cdfid; /* netcdf id */
|
|
int ia, id; /* attribute number */
|
|
nc_type type;
|
|
int len;
|
|
int vv_id; /* variable id */
|
|
static struct cdfvar vv = /* new variable */
|
|
{"vv", NC_SHORT, 2, ___, 0};
|
|
|
|
(void) fprintf(stderr, "*** Testing %s ...\t", &pname[5]);
|
|
|
|
if ((cdfid = ncopen(path, NC_WRITE)) == -1) {
|
|
error("%s: ncopen failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* in data mode, check all attributes against test netcdf */
|
|
for (ia = 0; ia < test.natts; ia++) {
|
|
if (ncattinq(cdfid, test.atts[ia].var, test.atts[ia].name,
|
|
&type, &len) == -1) {
|
|
error("%s: ncattinq failed", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
if (type != test.atts[ia].type) {
|
|
error("%s: ncattinq returned wrong type", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
if (len != test.atts[ia].len) {
|
|
error("%s: ncattinq returned wrong len", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
}
|
|
|
|
/* enter define mode */
|
|
if (ncredef(cdfid) == -1) {
|
|
error("%s: cdredef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in define mode, add a variable */
|
|
vv.dims = (int *) emalloc(sizeof(int) * vv.ndims);
|
|
for (id = 0; id < vv.ndims; id++)
|
|
vv.dims[id] = id; /* assumes vv.ndims <= test.ndims */
|
|
if ((vv_id = ncvardef(cdfid, vv.name, vv.type, vv.ndims, vv.dims))
|
|
== -1) {
|
|
error("%s: ncvardef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_var(&test, &vv); /* keep in-memory netcdf in sync */
|
|
|
|
/* try with nonexisting attribute, should fail */
|
|
if (ncattinq(cdfid, vv_id, "nonesuch", &type, &len) != -1) {
|
|
error("%s: ncattinq should fail with nonexisting attribute", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try with bad variable handle, should fail */
|
|
if (ncattinq(cdfid, test.nvars, test.atts[0].name, &type, &len) != -1) {
|
|
error("%s: ncattinq should fail with bad variable id", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in define mode check all attributes against test netcdf */
|
|
for (ia = 0; ia < test.natts; ia++) {
|
|
if (ncattinq(cdfid, test.atts[ia].var, test.atts[ia].name,
|
|
&type, &len) == -1) {
|
|
error("%s: ncattinq in define mode failed", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
if (type != test.atts[ia].type) {
|
|
error("%s: ncattinq in define mode returned wrong type", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
if (len != test.atts[ia].len) {
|
|
error("%s: ncattinq in define mode returned wrong len", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
}
|
|
if (ncendef (cdfid) == -1) {
|
|
error("%s: ncendef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncclose (cdfid) == -1) {
|
|
error("%s: ncclose failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
if (ncattinq(cdfid, NC_GLOBAL, test.atts[0].name, &type, &len) != -1) {
|
|
error("%s: ncattinq should fail with bad cdfid", pname);
|
|
nerrs++;
|
|
}
|
|
if (nerrs > 0)
|
|
(void) fprintf(stderr,"FAILED! ***\n");
|
|
else
|
|
(void) fprintf(stderr,"ok ***\n");
|
|
|
|
free(vv.dims);
|
|
return nerrs;
|
|
}
|
|
|
|
/*
|
|
* Test ncattget
|
|
* check that NC_GLOBAL variable id works
|
|
* check in both modes
|
|
* check that proper call worked after ncattput
|
|
* try with bad variable handle, check error
|
|
* try with nonexisting attribute, check error
|
|
* try with bad netCDF handle, check error
|
|
*/
|
|
int
|
|
test_ncattget(path)
|
|
const char *path; /* name of writable netcdf file to open */
|
|
{
|
|
int nerrs = 0;
|
|
int cdfid; /* netcdf id */
|
|
int ia, id;
|
|
static struct cdfatt tmp; /* attribute */
|
|
int uu_id; /* variable id */
|
|
static struct cdfvar uu = /* variable */
|
|
{"uu", NC_LONG, 2, ___, 0};
|
|
static nclong uumax = 1000; /* attribute value */
|
|
static struct cdfatt vmax = /* attribute */
|
|
{___, "valid_max", NC_LONG, 1, (void *) &uumax};
|
|
|
|
static char pname[] = "test_ncattget";
|
|
|
|
(void) fprintf(stderr, "*** Testing %s ...\t", &pname[5]);
|
|
|
|
if ((cdfid = ncopen(path, NC_WRITE)) == -1) {
|
|
error("%s: ncopen failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* enter define mode */
|
|
if (ncredef(cdfid) == -1) {
|
|
error("%s: cdredef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* add a variable */
|
|
uu.dims = (int *) emalloc(sizeof(int) * uu.ndims);
|
|
for (id = 0; id < uu.ndims; id++)
|
|
uu.dims[id] = id;
|
|
if ((uu_id = ncvardef(cdfid,
|
|
uu.name, uu.type, uu.ndims, uu.dims)) == -1) {
|
|
error("%s: ncvardef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_var(&test, &uu); /* keep in-memory netcdf in sync */
|
|
|
|
/* add an attribute */
|
|
if (ncattput(cdfid, uu_id,
|
|
vmax.name, vmax.type, vmax.len, vmax.val)
|
|
== -1) {
|
|
error("%s: ncattput of variable attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, uu_id, &vmax); /* keep in-memory netcdf updated */
|
|
|
|
/* in define mode, check all attributes values against test netcdf */
|
|
for (ia = 0; ia < test.natts; ia++) {
|
|
if (ncattinq(cdfid, test.atts[ia].var, test.atts[ia].name,
|
|
&tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq in define mode failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (tmp.type != test.atts[ia].type) {
|
|
error("%s: ncattinq in define mode returned wrong type", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
if (tmp.len != test.atts[ia].len) {
|
|
error("%s: ncattinq in define mode returned wrong len", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(tmp.len * nctypelen(tmp.type));
|
|
if (ncattget(cdfid, test.atts[ia].var, test.atts[ia].name, tmp.val)
|
|
== -1) {
|
|
error("%s: ncattget of variable attribute failed in define mode",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, test.atts[ia].val) != 0) {
|
|
error("%s: ncattget got bad values in define mode", pname);
|
|
error(" cdfid=%d, varname=%s, attname=%s, type=%d, len=%d",
|
|
cdfid, test.vars[test.atts[ia].var].name,
|
|
test.atts[ia].name, test.atts[ia].type, test.atts[ia].len);
|
|
(void)fprintf(stderr,"should have got:");
|
|
val_out(test.atts[ia].type, test.atts[ia].len,
|
|
test.atts[ia].val);
|
|
(void)fprintf(stderr," instead got:");
|
|
val_out(tmp.type, tmp.len, tmp.val);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
}
|
|
if (ncendef (cdfid) == -1) {
|
|
error("%s: ncendef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
|
|
/* in data mode, check all attributes values against test netcdf */
|
|
for (ia = 0; ia < test.natts; ia++) {
|
|
if (ncattinq(cdfid, test.atts[ia].var, test.atts[ia].name,
|
|
&tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq failed", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
if (tmp.type != test.atts[ia].type) {
|
|
error("%s: ncattinq returned wrong type", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
if (tmp.len != test.atts[ia].len) {
|
|
error("%s: ncattinq returned wrong len", pname);
|
|
ncclose(cdfid);
|
|
return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(tmp.len * nctypelen(tmp.type));
|
|
if (ncattget(cdfid, test.atts[ia].var, test.atts[ia].name, tmp.val)
|
|
== -1) {
|
|
error("%s: ncattget of variable attribute failed in data mode",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, test.atts[ia].val) != 0) {
|
|
error("%s: ncattget got bad values in data mode", pname);
|
|
error(" cdfid=%d, varname=%s, attname=%s, type=%d, len=%d",
|
|
cdfid, test.vars[test.atts[ia].var].name,
|
|
test.atts[ia].name, test.atts[ia].type, test.atts[ia].len);
|
|
(void)fprintf(stderr,"should have got:");
|
|
val_out(test.atts[ia].type, test.atts[ia].len,
|
|
test.atts[ia].val);
|
|
(void)fprintf(stderr," instead got:");
|
|
val_out(tmp.type, tmp.len, tmp.val);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
}
|
|
/* try with bad variable handle, should fail */
|
|
if (ncattget(cdfid, test.nvars, vmax.name, vmax.val) != -1) {
|
|
error("%s: ncattget should fail with bad variable handle", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try getting non-existent attribute, should fail */
|
|
if (ncattget(cdfid, uu_id, "nonesuch", vmax.val) != -1) {
|
|
error("%s: ncattget should fail with nonexistent attribute", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncclose (cdfid) == -1) {
|
|
error("%s: ncclose failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* try with bad netcdf handle, should fail */
|
|
if (ncattget(cdfid, uu_id, vmax.name, vmax.val) != -1) {
|
|
error("%s: ncattput should fail with bad netcdf handle", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (nerrs > 0)
|
|
(void) fprintf(stderr,"FAILED! ***\n");
|
|
else
|
|
(void) fprintf(stderr,"ok ***\n");
|
|
|
|
free(uu.dims);
|
|
return nerrs;
|
|
}
|
|
|
|
|
|
/*
|
|
* Test ncattcopy
|
|
* check that NC_GLOBAL variable for source or target works
|
|
* check that new attribute put works with target in define mode
|
|
* check that old attribute put works with target in data mode
|
|
* check that changing type and length of an attribute work OK
|
|
* try with same cdfid for source and target, different variables
|
|
* try with same cdfid for source and target, same variable
|
|
* try with nonexisting attribute, check error
|
|
* try with bad source or target netCDF handles, check error
|
|
* try with bad source or target variable handle, check error
|
|
*/
|
|
int
|
|
test_ncattcopy(path1, path2)
|
|
const char *path1; /* name of input netcdf file to open */
|
|
const char *path2; /* name of output netcdf file to create */
|
|
{
|
|
int nerrs = 0;
|
|
static char pname[] = "test_ncattcopy";
|
|
int cdfid, cdfid2; /* netcdf id */
|
|
int id; /* dimension id */
|
|
int tt_id; /* variable id */
|
|
static struct cdfvar tt = /* new variable for source netcdf */
|
|
{"tt", NC_LONG, 1, ___, 0};
|
|
int tu_id, tu2_id; /* variable ids */
|
|
static struct cdfvar tu = /* new variable for target netcdf */
|
|
{"tu", NC_DOUBLE, 2, ___, 0};
|
|
static double double_vals[] = {-1., -2.};
|
|
static float float_vals[] = {-1., -2.};
|
|
static struct cdfatt att = /* attribute */
|
|
{___, "att", NC_DOUBLE, LEN_OF(double_vals), (void *) double_vals};
|
|
static struct cdfatt att2 = /* attribute */
|
|
{___, "att", NC_FLOAT, LEN_OF(float_vals), (void *) float_vals};
|
|
static struct cdfatt tmp; /* attribute */
|
|
|
|
(void) fprintf(stderr, "*** Testing %s ...\t", &pname[5]);
|
|
|
|
if ((cdfid = ncopen(path1, NC_WRITE)) == -1) {
|
|
error("%s: ncopen failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* opened OK, enter define mode */
|
|
if (ncredef(cdfid) == -1) {
|
|
error("%s: ncredef failed on source", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in define mode, add a global attribute, a variable and an attribute */
|
|
if (ncattput(cdfid, NC_GLOBAL, att.name, att.type, att.len, att.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, NC_GLOBAL, &att); /* keep in-memory netcdf consistent */
|
|
tt.dims = (int *) emalloc(sizeof(int) * tt.ndims);
|
|
for (id=0; id < tt.ndims; id++)
|
|
tt.dims[0] = id;
|
|
if ((tt_id=ncvardef(cdfid, tt.name, tt.type, tt.ndims, tt.dims)) == -1) {
|
|
error("%s: ncvardef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_var(&test, &tt); /* keep in-memory netcdf consistent */
|
|
if (ncattput(cdfid, tt_id, att.name, att.type, att.len, att.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, tt_id, &att); /* keep in-memory netcdf consistent */
|
|
|
|
tu.dims = (int *) emalloc(sizeof(int) * tu.ndims);
|
|
for (id = 0; id < tu.ndims; id++)
|
|
tu.dims[id] = id;
|
|
if ((tu_id=ncvardef(cdfid, tu.name, tu.type, tu.ndims, tu.dims)) == -1) {
|
|
error("%s: ncvardef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_var(&test, &tu); /* keep in-memory netcdf consistent */
|
|
if (ncattput(cdfid, tu_id, att.name, att.type, att.len, att.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, tu_id, &att); /* keep in-memory netcdf consistent */
|
|
if (ncendef (cdfid) == -1) {
|
|
error("%s: ncendef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* first (source) netcdf is in data mode */
|
|
/* create second netCDF to copy attributes to */
|
|
if ((cdfid2 = nccreate(path2, NC_CLOBBER)) == -1) {
|
|
error("%s: nccreate failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* create dimensions and variable in second netcdf */
|
|
for (id = 0; id < tu.ndims; id++) { /* copy dimensions from source */
|
|
if ((tu.dims[id] =ncdimdef(cdfid2, test.dims[id].name,
|
|
test.dims[id].size)) == -1) {
|
|
error("%s: ncdimdef failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
}
|
|
if ((tu2_id=ncvardef(cdfid2, tu.name, tu.type, tu.ndims, tu.dims)) == -1) {
|
|
error("%s: ncvardef failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* try copying NC_GLOBAL attribute from source to target */
|
|
if (ncattcopy(cdfid, NC_GLOBAL, att.name, cdfid2, NC_GLOBAL) == -1) {
|
|
error("%s: ncattcopy on NC_GLOBAL attribute '%s' failed",
|
|
pname, att.name);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* check that copy worked with ncattinq and ncattget */
|
|
if (ncattinq(cdfid2, NC_GLOBAL, att.name, &tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq of NC_GLOBAL attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (att.type != tmp.type || att.len != tmp.len) {
|
|
error("%s: NC_GLOBAL ncattinq got unexpected type or len", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(att.len * nctypelen(att.type));
|
|
if (ncattget(cdfid2, NC_GLOBAL, att.name, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, att.val) != 0) {
|
|
error("%s: ncattget got bad values after put of NC_GLOBAL attrs",
|
|
pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
/* try copying variable attribute from source to target */
|
|
if (ncattcopy(cdfid, tt_id, att.name, cdfid2, tu2_id) == -1) {
|
|
error("%s: ncattcopy failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* check that copy worked with ncattinq and ncattget */
|
|
if (ncattinq(cdfid2, tu2_id, att.name, &tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq of variable attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (att.type != tmp.type || att.len != tmp.len) {
|
|
error("%s: variable ncattinq got unexpected type or len", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(att.len * nctypelen(att.type));
|
|
if (ncattget(cdfid2, tu2_id, att.name, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, att.val) != 0) {
|
|
error("%s: ncattget got bad values after copy of variable attrs",
|
|
pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
|
|
/*
|
|
* check that old attribute put works with target in data mode,
|
|
* also checks that changing type and length of an attribute works OK
|
|
*/
|
|
if (ncendef (cdfid2) == -1) {
|
|
error("%s: ncendef failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* change attribute to shorter attribute */
|
|
if (ncattput(cdfid, NC_GLOBAL, att2.name, att2.type, att2.len, att2.val)
|
|
== -1) {
|
|
error("%s: ncattput of shorter NC_GLOBAL attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
add_att(&test, NC_GLOBAL, &att2); /* keep in-memory netcdf consistent */
|
|
/* copy shorter attribute on existing attribute */
|
|
if (ncattcopy(cdfid, NC_GLOBAL, att2.name, cdfid2, tu2_id) == -1) {
|
|
error("%s: ncattcopy of shorter attribute on old attribute failed",
|
|
pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* check that copy worked with ncattinq and ncattget */
|
|
if (ncattinq(cdfid2, tu2_id, att2.name, &tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq of variable attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (att2.type != tmp.type || att2.len != tmp.len) {
|
|
error("%s: variable ncattinq got unexpected type or len", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(att2.len * nctypelen(att2.type));
|
|
if (ncattget(cdfid2, tu2_id, att2.name, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, att2.val) != 0) {
|
|
error("%s: ncattget got bad values after copy of variable attrs",
|
|
pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
|
|
/* try copying with same source and target netcdf, different variables */
|
|
/* copy shorter attribute on existing attribute */
|
|
if (ncattcopy(cdfid, NC_GLOBAL, att2.name, cdfid, tu_id) == -1) {
|
|
error("%s: ncattcopy of shorter NC_GLOBAL attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
add_att(&test, tu_id, &att2); /* keep in-memory netcdf consistent */
|
|
/* check that copy worked with ncattinq and ncattget */
|
|
if (ncattinq(cdfid, tu_id, att2.name, &tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq of variable attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (att2.type != tmp.type || att2.len != tmp.len) {
|
|
error("%s: variable ncattinq got unexpected type or len", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(att2.len * nctypelen(att2.type));
|
|
if (ncattget(cdfid, tu_id, att2.name, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed", pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, att2.val) != 0) {
|
|
error("%s: ncattget got bad values after copy of variable attrs",
|
|
pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.val);
|
|
|
|
/* try with same cdfid for source and target, same variable */
|
|
if (ncattcopy(cdfid, tu_id, att.name, cdfid, tu_id) == -1) {
|
|
error("%s: ncattcopy failed with identical source and target",
|
|
pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* try with nonexisting attribute, check error */
|
|
if (ncattcopy(cdfid, tt_id, "nonesuch", cdfid, tu_id) != -1) {
|
|
error("%s: ncattcopy should fail with bad attribute name",
|
|
pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
/* try with bad source or target variable handle, check error */
|
|
if (ncattcopy(cdfid, test.nvars, att.name, cdfid, tu_id) != -1) {
|
|
error("%s: ncattcopy should fail with bad source variable id",
|
|
pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (ncattcopy(cdfid, tt_id, att.name, cdfid, 2) != -1) {
|
|
error("%s: ncattcopy should fail with bad target variable id",
|
|
pname);
|
|
ncclose(cdfid); ncclose(cdfid2); return ++nerrs;
|
|
}
|
|
if (ncclose (cdfid2) == -1) {
|
|
error("%s: ncclose failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try with bad source or target netCDF handles, check error */
|
|
if (ncattcopy(cdfid, tt_id, att.name, cdfid2, tu_id) != -1) {
|
|
error("%s: ncattcopy should fail with bad target netcdf id",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncclose (cdfid) == -1) {
|
|
error("%s: ncclose failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
if (ncattcopy(cdfid, tt_id, att.name, cdfid2, tu_id) != -1) {
|
|
error("%s: ncattcopy should fail with bad netcdf id", pname);
|
|
nerrs++;
|
|
}
|
|
if (nerrs > 0)
|
|
(void) fprintf(stderr,"FAILED! ***\n");
|
|
else
|
|
(void) fprintf(stderr,"ok ***\n");
|
|
|
|
free(tt.dims);
|
|
free(tu.dims);
|
|
|
|
return nerrs;
|
|
}
|
|
|
|
|
|
/*
|
|
* Test ncattname
|
|
* check that NC_GLOBAL variable id works
|
|
* check in both modes
|
|
* check that proper call worked after ncattput
|
|
* try with bad netCDF handle, check error
|
|
* try with bad variable handle, check error
|
|
* try with bad attribute number, check error
|
|
*/
|
|
int
|
|
test_ncattname(path)
|
|
const char *path; /* name of writable netcdf file to open */
|
|
{
|
|
int nerrs = 0;
|
|
static char pname[] = "test_ncattname";
|
|
int cdfid; /* netcdf id */
|
|
struct cdfatt tmp; /* attributes */
|
|
int ia, ib; /* attribute numbers */
|
|
int iv; /* variable id */
|
|
static short short_vals[] = {3, 4, 5};
|
|
static struct cdfatt att = /* attribute */
|
|
{___, ___, NC_SHORT, LEN_OF(short_vals), (void *) short_vals};
|
|
|
|
(void) fprintf(stderr, "*** Testing %s ...\t", &pname[5]);
|
|
|
|
if ((cdfid = ncopen(path, NC_WRITE)) == -1) {
|
|
error("%s: ncopen failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* opened OK, enter define mode */
|
|
if (ncredef(cdfid) == -1) {
|
|
error("%s: ncredef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* for each NC_GLOBAL attribute, get name and compare with expected name */
|
|
att.name = (char *) emalloc(MAX_NC_NAME);
|
|
ib = 0;
|
|
for (ia = 0; ia < test.ngatts; ia++) {
|
|
if (ncattname(cdfid, NC_GLOBAL, ia, att.name) == -1) {
|
|
error("%s: ncattname failed on global attribute", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* find number of next global attribute */
|
|
while (ib < test.natts && test.atts[ib].var != NC_GLOBAL)
|
|
ib++;
|
|
if (ib >= test.natts) {
|
|
error("%s: test problem, expected global attribute not found",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (strcmp(att.name, test.atts[ib].name) != 0) {
|
|
error("%s: NC_GLOBAL attribute name `%s' instead of expected `%s'",
|
|
pname, att.name, test.atts[ib].name);
|
|
nerrs++;
|
|
}
|
|
ib++;
|
|
}
|
|
/* for each variable attribute, get name and compare with expected name */
|
|
for (iv = 0; iv < test.nvars; iv++) {
|
|
ib = 0;
|
|
for (ia = 0; ia < test.vars[iv].natts; ia++) {
|
|
if (ncattname(cdfid, iv, ia, att.name) == -1) {
|
|
error("%s: ncattname failed on variable attribute", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* find number of next attribute */
|
|
while (ib < test.natts && test.atts[ib].var != iv)
|
|
ib++;
|
|
if (ib >= test.natts) {
|
|
error("%s: problem in test, expected attribute not found",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (strcmp(att.name, test.atts[ib].name) != 0) {
|
|
error("%s: variable '%s' name `%s' instead of expected `%s'",
|
|
pname, test.vars[iv].name, att.name, test.atts[ib].name);
|
|
nerrs++;
|
|
}
|
|
ib++;
|
|
}
|
|
}
|
|
/* in define mode, add a global attribute */
|
|
(void) strcpy(att.name,"attx");
|
|
if (ncattput(cdfid, NC_GLOBAL, att.name, att.type, att.len, att.val)
|
|
== -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, NC_GLOBAL, &att); /* keep in-memory netcdf consistent */
|
|
/* test that ncattname works immediately after ncattput */
|
|
tmp.name = (char *) emalloc(MAX_NC_NAME);
|
|
if (ncattname(cdfid, NC_GLOBAL, test.ngatts-1, tmp.name) == -1) {
|
|
error("%s: ncattname failed on variable attribute", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (strcmp(att.name, tmp.name) != 0) {
|
|
error("%s: immediate NC_GLOBAL name `%s' instead of expected `%s'",
|
|
pname, tmp.name, att.name);
|
|
nerrs++;
|
|
}
|
|
if (ncendef (cdfid) == -1) {
|
|
error("%s: ncendef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in data mode */
|
|
/* for each NC_GLOBAL attribute, get name and compare with expected name */
|
|
ib = 0;
|
|
for (ia = 0; ia < test.ngatts; ia++) {
|
|
if (ncattname(cdfid, NC_GLOBAL, ia, att.name) == -1) {
|
|
error("%s: ncattname failed on global attribute", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* find number of next global attribute */
|
|
while (ib < test.natts && test.atts[ib].var != NC_GLOBAL)
|
|
ib++;
|
|
if (ib >= test.natts) {
|
|
error("%s: test problem, expected global attribute not found",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (strcmp(att.name, test.atts[ib].name) != 0) {
|
|
error("%s: NC_GLOBAL attribute name `%s' instead of expected `%s'",
|
|
pname, att.name, test.atts[ib].name);
|
|
nerrs++;
|
|
}
|
|
ib++;
|
|
}
|
|
/* for each variable attribute, get name and compare with expected name */
|
|
for (iv = 0; iv < test.nvars; iv++) {
|
|
ib = 0;
|
|
for (ia = 0; ia < test.vars[iv].natts; ia++) {
|
|
if (ncattname(cdfid, iv, ia, att.name) == -1) {
|
|
error("%s: ncattname failed on variable attribute", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* find number of next attribute */
|
|
while (ib < test.natts && test.atts[ib].var != iv)
|
|
ib++;
|
|
if (ib >= test.natts) {
|
|
error("%s: problem in test, expected attribute not found",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (strcmp(att.name, test.atts[ib].name) != 0) {
|
|
error("%s: variable '%s' name `%s' instead of expected `%s'",
|
|
pname, test.vars[iv].name, att.name, test.atts[ib].name);
|
|
nerrs++;
|
|
}
|
|
ib++;
|
|
}
|
|
}
|
|
/* try with bad variable handle, check error */
|
|
if (ncattname(cdfid, test.nvars, 0, att.name) != -1) {
|
|
error("%s: ncattname should fail with bad variable handle", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try with bad attribute number, check error */
|
|
if (ncattname(cdfid, NC_GLOBAL, -1, att.name) != -1) {
|
|
error("%s: ncattname should fail with negative number", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncattname(cdfid, NC_GLOBAL, test.ngatts, att.name) != -1) {
|
|
error("%s: ncattname should fail with too-high number", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncclose (cdfid) == -1) {
|
|
error("%s: ncclose failed", pname);
|
|
nerrs++;
|
|
return ++nerrs;
|
|
}
|
|
/* try with bad netCDF handle, check error */
|
|
if (ncattname(cdfid, NC_GLOBAL, 0, att.name) != -1) {
|
|
error("%s: ncattname shoul fail with bad cdfid", pname);
|
|
nerrs++;
|
|
}
|
|
free (tmp.name);
|
|
free (att.name);
|
|
if (nerrs > 0)
|
|
(void) fprintf(stderr,"FAILED! ***\n");
|
|
else
|
|
(void) fprintf(stderr,"ok ***\n");
|
|
|
|
return nerrs;
|
|
}
|
|
|
|
|
|
/*
|
|
* Test ncattrename
|
|
* check that proper rename worked with ncattinq, ncattget
|
|
* try renaming to existing attribute name, check error
|
|
* try with nonexisting attribute, check error
|
|
* try with bad variable handle, check error
|
|
* try in data mode, check error
|
|
* try with bad netCDF handle, check error
|
|
*/
|
|
int
|
|
test_ncattrename(path)
|
|
const char *path; /* name of writable netcdf file to open */
|
|
{
|
|
int nerrs = 0;
|
|
static char pname[] = "test_ncattrename";
|
|
int cdfid; /* netcdf id */
|
|
static char newname[] = "shorter";
|
|
static char longername[] = "longer_name";
|
|
struct cdfatt tmp; /* attributes */
|
|
static short short_vals[] = {3, 4, 5};
|
|
static struct cdfatt atty = /* attribute */
|
|
{___, "long_name", NC_SHORT, LEN_OF(short_vals), (void *) short_vals};
|
|
static struct cdfatt attz = /* attribute */
|
|
{___, "arggh", NC_SHORT, LEN_OF(short_vals), (void *) short_vals};
|
|
int ynum; /* attribute number */
|
|
|
|
(void) fprintf(stderr, "*** Testing %s ...\t", &pname[5]);
|
|
|
|
if ((cdfid = ncopen(path, NC_WRITE)) == -1) {
|
|
error("%s: ncopen failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* opened OK, enter define mode */
|
|
if (ncredef(cdfid) == -1) {
|
|
error("%s: cdredef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in define mode, add two attributes */
|
|
if (ncattput(cdfid, NC_GLOBAL, atty.name, atty.type, atty.len,
|
|
atty.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, NC_GLOBAL, &atty); /* keep in-memory netcdf in sync */
|
|
ynum = test.natts-1; /* number of attribute just put */
|
|
if (ncattput(cdfid, NC_GLOBAL, attz.name, attz.type, attz.len,
|
|
attz.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, NC_GLOBAL, &attz); /* keep in-memory netcdf in sync */
|
|
|
|
/* rename first attribute to shorter name */
|
|
if (ncattrename(cdfid, NC_GLOBAL, atty.name, newname) == -1) {
|
|
error("%s: ncattrename failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
(void) strcpy(test.atts[ynum].name, newname); /* keep test consistent */
|
|
/* check new name with ncattinq */
|
|
if (ncattinq(cdfid, NC_GLOBAL, newname, &tmp.type, &tmp.len) == -1) {
|
|
error("%s: ncattinq of renamed attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (atty.type != tmp.type || atty.len != tmp.len) {
|
|
error("%s: NC_GLOBAL ncattinq got unexpected type or len", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* allocate space to hold the attribute value to be retrieved */
|
|
tmp.val = emalloc(atty.len * nctypelen(atty.type));
|
|
if (ncattget(cdfid, NC_GLOBAL, newname, tmp.val) == -1) {
|
|
error("%s: ncattget of variable attribute failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, atty.val) != 0) {
|
|
error("%s: ncattget got bad values after rename attrs", pname);
|
|
nerrs++;
|
|
return ++nerrs;
|
|
}
|
|
if (ncattinq(cdfid, NC_GLOBAL, atty.name, &tmp.type, &tmp.len) != -1) {
|
|
error("%s: ncattrename left attribute with old name", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try to rename second attribute same as first, should fail */
|
|
if (ncattrename(cdfid, NC_GLOBAL, attz.name, newname) != -1) {
|
|
error("%s: ncattrename should have failed with used name", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try to rename second attribute with a longer name */
|
|
if (ncattrename(cdfid, NC_GLOBAL, attz.name, longername) == -1) {
|
|
error("%s: ncattrename failed with longer name", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try with bad variable handle, check for failure */
|
|
if (ncattrename(cdfid, test.nvars, newname, atty.name) != -1) {
|
|
error("%s: ncattrename should have failed on bad variable id", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try with bad attribute name, check for failure */
|
|
if (ncattrename(cdfid, NC_GLOBAL, "nonesuch", newname) != -1) {
|
|
error("%s: ncattrename should have failed on bad attribute name",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncendef (cdfid) == -1) {
|
|
error("%s: ncendef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in data mode change name to even shorter and check value */
|
|
if (ncattrename(cdfid, NC_GLOBAL, newname, "short") == -1) {
|
|
error("%s: ncattrename to shorter name failed in data mode", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncattrename(cdfid, NC_GLOBAL, "short", "plugh") == -1) {
|
|
error("%s: ncattrename to same length failed in data mode", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncattget(cdfid, NC_GLOBAL, "plugh", tmp.val) == -1) {
|
|
error("%s: ncgetatt of renamed attribute failed in data mode", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (val_cmp(tmp.type, tmp.len, tmp.val, atty.val) != 0) {
|
|
error("%s: ncattget got bad values after data mode rename", pname);
|
|
nerrs++;
|
|
return ++nerrs;
|
|
}
|
|
free (tmp.val);
|
|
if (ncclose (cdfid) == -1) {
|
|
error("%s: ncclose failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* should fail, since bad handle */
|
|
if (ncattrename(cdfid, NC_GLOBAL, newname, atty.name) != -1) {
|
|
error("%s: ncattrename should fail with bad cdfid", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (nerrs > 0)
|
|
(void) fprintf(stderr,"FAILED! ***\n");
|
|
else
|
|
(void) fprintf(stderr,"ok ***\n");
|
|
|
|
return nerrs;
|
|
}
|
|
|
|
|
|
/*
|
|
* Test ncattdel
|
|
* check that proper delete worked
|
|
* try with bad netCDF handle, check error
|
|
* try with bad variable handle, check error
|
|
* try with nonexisting attribute, check error
|
|
* try in data mode, check error
|
|
*/
|
|
int
|
|
test_ncattdel(path)
|
|
const char *path; /* name of writable netcdf file to open */
|
|
{
|
|
int nerrs = 0;
|
|
static char pname[] = "test_ncattdel";
|
|
int cdfid; /* netcdf id */
|
|
static short short_vals[] = {-1, -2, -3 };
|
|
static struct cdfatt yaa = /* attribute */
|
|
{___, "yet_another_attribute", NC_SHORT, LEN_OF(short_vals),
|
|
(void *) short_vals};
|
|
int id; /* dimension id */
|
|
int yav_id; /* variable id */
|
|
static struct cdfvar yav = /* new variable for target netcdf */
|
|
{"yet_another_variable", NC_DOUBLE, 2, ___, 0};
|
|
struct cdfvar vtmp; /* variable */
|
|
struct cdfatt atmp; /* attribute */
|
|
int ndims; /* number of dimensions */
|
|
int nvars; /* number of variables */
|
|
int ngatts1, ngatts2; /* number of global attributes */
|
|
int natts; /* number of variable attributes */
|
|
int xdimid; /* id of unlimited dimension */
|
|
|
|
(void) fprintf(stderr, "*** Testing %s ...\t", &pname[5]);
|
|
|
|
if ((cdfid = ncopen(path, NC_WRITE)) == -1) {
|
|
error("%s: ncopen failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* opened OK, enter define mode */
|
|
if (ncredef(cdfid) == -1) {
|
|
error("%s: cdredef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in define mode, add global attribute, variable, variable attribute */
|
|
if (ncattput(cdfid, NC_GLOBAL, yaa.name, yaa.type, yaa.len, yaa.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, NC_GLOBAL, &yaa); /* keep in-memory netcdf in sync */
|
|
yav.dims = (int *) emalloc(sizeof(int) * yav.ndims);
|
|
for (id = 0; id < yav.ndims; id++)
|
|
yav.dims[id] = id;
|
|
if ((yav_id=ncvardef(cdfid, yav.name, yav.type, yav.ndims, yav.dims))
|
|
== -1) {
|
|
error("%s: ncvardef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_var(&test, &yav); /* keep in-memory netcdf consistent */
|
|
if (ncattput(cdfid, yav_id, yaa.name, yaa.type, yaa.len, yaa.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, yav_id, &yaa); /* keep in-memory netcdf consistent */
|
|
|
|
/* get number of global attributes, number of attributes for variable */
|
|
if (ncinquire(cdfid, &ndims, &nvars, &ngatts1, &xdimid) == -1) {
|
|
error("%s: ncinquire in data mode failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
vtmp.dims = (int *) emalloc(sizeof(int) * NC_MAX_VAR_DIMS);
|
|
vtmp.name = (char *) emalloc(MAX_NC_NAME);
|
|
if (ncvarinq(cdfid, yav_id, vtmp.name, &vtmp.type, &vtmp.ndims, vtmp.dims,
|
|
&natts) == -1) {
|
|
error("%s: ncvarinq failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
|
|
/* delete global attribute and check that it's gone */
|
|
if (ncattdel(cdfid, NC_GLOBAL, yaa.name) == -1) {
|
|
error("%s: ncattdel failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
del_att(&test, NC_GLOBAL, &yaa); /* keep in-memory netcdf consistent */
|
|
if (ncinquire(cdfid, &ndims, &nvars, &ngatts2, &xdimid) == -1) {
|
|
error("%s: ncinquire failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ngatts2 != ngatts1 - 1) {
|
|
error("%s: NC_GLOBAL attribute deleted, but ngatts did not decrement",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncattinq(cdfid, NC_GLOBAL, yaa.name, &atmp.type, &atmp.len) != -1) {
|
|
error("%s: ncattinq on deleted NC_GLOBAL attribute should fail", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
|
|
/* delete variable attribute and check that it's gone */
|
|
if (ncattdel(cdfid, yav_id, yaa.name) == -1) {
|
|
error("%s: ncattdel failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
del_att(&test, yav_id, &yaa); /* keep in-memory netcdf consistent */
|
|
if (ncvarinq(cdfid, yav_id, vtmp.name, &vtmp.type, &vtmp.ndims,
|
|
vtmp.dims, &vtmp.natts) == -1) {
|
|
error("%s: ncvarinq failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (vtmp.natts != natts - 1) {
|
|
error("%s: NC_GLOBAL attribute deleted, but ngatts did not decrement",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncattinq(cdfid, yav_id, yaa.name, &atmp.type, &atmp.len) != -1) {
|
|
error("%s: ncattinq on deleted variable attribute should fail",
|
|
pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* re-add global attribute, variable, variable attribute */
|
|
if (ncattput(cdfid, NC_GLOBAL, yaa.name, yaa.type, yaa.len, yaa.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, NC_GLOBAL, &yaa); /* keep in-memory netcdf in sync */
|
|
if (ncattput(cdfid, yav_id, yaa.name, yaa.type, yaa.len, yaa.val) == -1) {
|
|
error("%s: ncattput failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
add_att(&test, yav_id, &yaa); /* keep in-memory netcdf consistent */
|
|
/* try on nonexistent attribute, should fail */
|
|
if (ncattdel(cdfid, yav_id, "nonesuch") != -1) {
|
|
error("%s: ncattdel should fail on bogus attribute", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* try on bad variable id, should fail */
|
|
if (ncattdel(cdfid, test.nvars, yaa.name) != -1) {
|
|
error("%s: ncattdel should fail on bad variable id", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncendef (cdfid) == -1) {
|
|
error("%s: ncendef failed", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
/* in data mode, should fail */
|
|
if (ncattdel(cdfid, NC_GLOBAL, yaa.name) != -1) {
|
|
error("%s: ncattdel in data mode should fail", pname);
|
|
ncclose(cdfid); return ++nerrs;
|
|
}
|
|
if (ncclose (cdfid) == -1) {
|
|
error("%s: ncclose failed", pname);
|
|
return ++nerrs;
|
|
}
|
|
/* try on bad netcdf handle, should fail */
|
|
if (ncattdel(cdfid, yav_id, yaa.name) != -1) {
|
|
error("%s: ncattdel should fail on bad netcdf id", pname);
|
|
nerrs++;
|
|
}
|
|
free(vtmp.dims);
|
|
free(vtmp.name);
|
|
free(yav.dims);
|
|
if (nerrs > 0)
|
|
(void) fprintf(stderr,"FAILED! ***\n");
|
|
else
|
|
(void) fprintf(stderr,"ok ***\n");
|
|
|
|
return nerrs;
|
|
}
|