2018-12-07 06:47:47 +08:00
|
|
|
/* Copyright 2018, UCAR/Unidata and OPeNDAP, Inc.
|
2012-08-01 04:34:13 +08:00
|
|
|
See the COPYRIGHT file for more information. */
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
#include "ocinternal.h"
|
|
|
|
#include "occompile.h"
|
|
|
|
#include "ocdebug.h"
|
|
|
|
|
2015-11-30 11:38:12 +08:00
|
|
|
static OCerror mergedas1(OCnode* dds, OCnode* das);
|
|
|
|
static OCerror mergedods1(OCnode* dds, OCnode* das);
|
|
|
|
static OCerror mergeother1(OCnode* root, OCnode* das);
|
2017-03-09 08:01:10 +08:00
|
|
|
static char* pathtostring(NClist* path, char* separator);
|
2012-08-01 04:34:13 +08:00
|
|
|
static void computefullname(OCnode* node);
|
2017-03-12 06:20:20 +08:00
|
|
|
static OCerror mergeother1(OCnode* root, OCnode* das);
|
2017-03-18 07:20:02 +08:00
|
|
|
static OCerror mergeother(OCnode* ddsroot, NClist* dasnodes);
|
2012-08-01 04:34:13 +08:00
|
|
|
|
|
|
|
/* Process ocnodes to fix various semantic issues*/
|
|
|
|
void
|
2017-03-09 08:01:10 +08:00
|
|
|
occomputesemantics(NClist* ocnodes)
|
2012-08-01 04:34:13 +08:00
|
|
|
{
|
|
|
|
unsigned int i,j;
|
|
|
|
OCASSERT((ocnodes != NULL));
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(ocnodes);i++) {
|
|
|
|
OCnode* node = (OCnode*)nclistget(ocnodes,i);
|
2012-08-01 04:34:13 +08:00
|
|
|
/* set the container for dims*/
|
|
|
|
if(node->octype == OC_Dimension && node->dim.array != NULL) {
|
|
|
|
node->container = node->dim.array->container;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Fill in array.sizes */
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(ocnodes);i++) {
|
|
|
|
OCnode* node = (OCnode*)nclistget(ocnodes,i);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(node->array.rank > 0) {
|
|
|
|
node->array.sizes = (size_t*)malloc(node->array.rank*sizeof(size_t));
|
|
|
|
for(j=0;j<node->array.rank;j++) {
|
2017-03-09 08:01:10 +08:00
|
|
|
OCnode* dim = (OCnode*)nclistget(node->array.dimensions,j);
|
2012-08-01 04:34:13 +08:00
|
|
|
node->array.sizes[j] = dim->dim.declsize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
occomputefullnames(OCnode* root)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
if(root->name != NULL) computefullname(root);
|
|
|
|
if(root->subnodes != NULL) { /* recurse*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(root->subnodes);i++) {
|
|
|
|
OCnode* node = (OCnode*)nclistget(root->subnodes,i);
|
2012-08-01 04:34:13 +08:00
|
|
|
occomputefullnames(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
computefullname(OCnode* node)
|
|
|
|
{
|
|
|
|
char* tmp;
|
|
|
|
char* fullname;
|
2017-03-09 08:01:10 +08:00
|
|
|
NClist* path;
|
2012-08-01 04:34:13 +08:00
|
|
|
|
|
|
|
OCASSERT((node->name != NULL));
|
2015-11-30 11:38:12 +08:00
|
|
|
if(node->fullname != NULL)
|
|
|
|
return;
|
2017-03-09 08:01:10 +08:00
|
|
|
path = nclistnew();
|
2012-08-01 04:34:13 +08:00
|
|
|
occollectpathtonode(node,path);
|
2015-11-30 11:38:12 +08:00
|
|
|
tmp = pathtostring(path,PATHSEPARATOR);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(tmp == NULL) {
|
|
|
|
fullname = nulldup(node->name);
|
|
|
|
} else {
|
|
|
|
fullname = tmp;
|
|
|
|
}
|
|
|
|
node->fullname = fullname;
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistfree(path);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
|
2015-11-30 11:38:12 +08:00
|
|
|
/* Convert path to a string */
|
2012-08-01 04:34:13 +08:00
|
|
|
static char*
|
2017-03-09 08:01:10 +08:00
|
|
|
pathtostring(NClist* path, char* separator)
|
2012-08-01 04:34:13 +08:00
|
|
|
{
|
|
|
|
int slen,i,len;
|
|
|
|
char* pathname;
|
2015-11-30 11:38:12 +08:00
|
|
|
if(path == NULL) return NULL;
|
2017-03-09 08:01:10 +08:00
|
|
|
len = nclistlength(path);
|
2015-11-30 11:38:12 +08:00
|
|
|
if(len == 0) return NULL;
|
|
|
|
for(i=0,slen=0;i<len;i++) {
|
2017-03-09 08:01:10 +08:00
|
|
|
OCnode* node = (OCnode*)nclistget(path,(size_t)i);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(node->container == NULL || node->name == NULL) continue;
|
|
|
|
slen += strlen(node->name);
|
|
|
|
}
|
|
|
|
slen += ((len-1)*strlen(separator));
|
|
|
|
slen += 1; /* for null terminator*/
|
2014-03-09 11:41:30 +08:00
|
|
|
pathname = (char*)ocmalloc((size_t)slen);
|
2012-08-01 04:34:13 +08:00
|
|
|
MEMCHECK(pathname,NULL);
|
|
|
|
pathname[0] = '\0';
|
|
|
|
for(i=0;i<len;i++) {
|
2017-03-09 08:01:10 +08:00
|
|
|
OCnode* node = (OCnode*)nclistget(path,(size_t)i);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(node->container == NULL || node->name == NULL) continue;
|
|
|
|
if(strlen(pathname) > 0) strcat(pathname,separator);
|
|
|
|
strcat(pathname,node->name);
|
|
|
|
}
|
|
|
|
return pathname;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Collect the set of nodes ending in "node"*/
|
|
|
|
void
|
2017-03-09 08:01:10 +08:00
|
|
|
occollectpathtonode(OCnode* node, NClist* path)
|
2012-08-01 04:34:13 +08:00
|
|
|
{
|
|
|
|
if(node == NULL) return;
|
|
|
|
occollectpathtonode(node->container,path);
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(path,(void*)node);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
OCnode*
|
|
|
|
ocnode_new(char* name, OCtype ptype, OCnode* root)
|
|
|
|
{
|
|
|
|
OCnode* cdf = (OCnode*)ocmalloc(sizeof(OCnode));
|
|
|
|
MEMCHECK(cdf,(OCnode*)NULL);
|
|
|
|
memset((void*)cdf,0,sizeof(OCnode));
|
|
|
|
cdf->header.magic = OCMAGIC;
|
|
|
|
cdf->header.occlass = OC_Node;
|
|
|
|
cdf->name = (name?nulldup(name):NULL);
|
|
|
|
cdf->octype = ptype;
|
|
|
|
cdf->array.dimensions = NULL;
|
|
|
|
cdf->root = root;
|
|
|
|
return cdf;
|
|
|
|
}
|
|
|
|
|
2013-03-25 01:33:17 +08:00
|
|
|
static OCattribute*
|
2017-03-09 08:01:10 +08:00
|
|
|
makeattribute(char* name, OCtype ptype, NClist* values)
|
2012-08-01 04:34:13 +08:00
|
|
|
{
|
|
|
|
OCattribute* att = (OCattribute*)ocmalloc(sizeof(OCattribute)); /* ocmalloc zeros*/
|
|
|
|
MEMCHECK(att,(OCattribute*)NULL);
|
|
|
|
att->name = nulldup(name);
|
|
|
|
att->etype = ptype;
|
2017-03-09 08:01:10 +08:00
|
|
|
att->nvalues = nclistlength(values);
|
2012-08-01 04:34:13 +08:00
|
|
|
att->values = NULL;
|
|
|
|
if(att->nvalues > 0) {
|
|
|
|
int i;
|
|
|
|
att->values = (char**)ocmalloc(sizeof(char*)*att->nvalues);
|
|
|
|
for(i=0;i<att->nvalues;i++)
|
2017-03-09 08:01:10 +08:00
|
|
|
att->values[i] = nulldup((char*)nclistget(values,(size_t)i));
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
return att;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ocroot_free(OCnode* root)
|
|
|
|
{
|
|
|
|
OCtree* tree;
|
|
|
|
OCstate* state;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if(root == NULL || root->tree == NULL) return;
|
|
|
|
|
|
|
|
tree = root->tree;
|
|
|
|
state = tree->state;
|
|
|
|
|
|
|
|
/* Free up the OCDATA instance, if any */
|
|
|
|
if(tree->data.data != NULL)
|
|
|
|
ocdata_free(state,tree->data.data);
|
|
|
|
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(state->trees);i++) {
|
|
|
|
OCnode* node = (OCnode*)nclistget(state->trees,(size_t)i);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(root == node)
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistremove(state->trees,(size_t)i);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
/* Note: it is ok if state->trees does not contain this root */
|
|
|
|
octree_free(tree);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
octree_free(OCtree* tree)
|
|
|
|
{
|
|
|
|
if(tree == NULL) return;
|
|
|
|
ocnodes_free(tree->nodes);
|
|
|
|
ocfree(tree->constraint);
|
|
|
|
ocfree(tree->text);
|
|
|
|
if(tree->data.xdrs != NULL) {
|
|
|
|
xxdr_free(tree->data.xdrs);
|
|
|
|
}
|
|
|
|
ocfree(tree->data.filename); /* may be null */
|
|
|
|
if(tree->data.file != NULL) fclose(tree->data.file);
|
|
|
|
ocfree(tree->data.memory);
|
|
|
|
ocfree(tree);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2017-03-09 08:01:10 +08:00
|
|
|
ocnodes_free(NClist* nodes)
|
2012-08-01 04:34:13 +08:00
|
|
|
{
|
|
|
|
unsigned int i,j;
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(nodes);i++) {
|
|
|
|
OCnode* node = (OCnode*)nclistget(nodes,i);
|
2012-08-01 04:34:13 +08:00
|
|
|
ocfree(node->name);
|
|
|
|
ocfree(node->fullname);
|
2017-03-09 08:01:10 +08:00
|
|
|
while(nclistlength(node->att.values) > 0) {
|
|
|
|
char* value = (char*)nclistpop(node->att.values);
|
2012-08-01 04:34:13 +08:00
|
|
|
ocfree(value);
|
|
|
|
}
|
2017-03-09 08:01:10 +08:00
|
|
|
while(nclistlength(node->attributes) > 0) {
|
|
|
|
OCattribute* attr = (OCattribute*)nclistpop(node->attributes);
|
2012-08-01 04:34:13 +08:00
|
|
|
ocfree(attr->name);
|
2012-08-03 02:43:21 +08:00
|
|
|
#if 0
|
2012-08-01 04:34:13 +08:00
|
|
|
/* If the attribute type is string, then we need to free them*/
|
2012-08-03 02:43:21 +08:00
|
|
|
all values are strings now
|
|
|
|
if(attr->etype == OC_String || attr->etype == OC_URL)
|
|
|
|
#endif
|
|
|
|
{
|
2012-08-01 04:34:13 +08:00
|
|
|
char** strings = (char**)attr->values;
|
|
|
|
for(j=0;j<attr->nvalues;j++) {ocfree(*strings); strings++;}
|
|
|
|
}
|
|
|
|
ocfree(attr->values);
|
|
|
|
ocfree(attr);
|
|
|
|
}
|
2017-03-09 08:01:10 +08:00
|
|
|
if(node->array.dimensions != NULL) nclistfree(node->array.dimensions);
|
|
|
|
if(node->subnodes != NULL) nclistfree(node->subnodes);
|
|
|
|
if(node->att.values != NULL) nclistfree(node->att.values);
|
|
|
|
if(node->attributes != NULL) nclistfree(node->attributes);
|
2012-08-03 02:43:21 +08:00
|
|
|
if(node->array.sizes != NULL) free(node->array.sizes);
|
2012-08-01 04:34:13 +08:00
|
|
|
ocfree(node);
|
|
|
|
}
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistfree(nodes);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
In order to be as compatible as possible with libdap,
|
|
|
|
we try to use the same algorithm for DAS->DDS matching.
|
|
|
|
As described there, the algorithm is as follows.
|
|
|
|
If the [attribute] name contains one or
|
|
|
|
more field separators then look for a [DDS]variable whose
|
|
|
|
name matches exactly. If the name contains no field separators then
|
|
|
|
the look first in the top level [of the DDS] and then in all subsequent
|
|
|
|
levels and return the first occurrence found. In general, this
|
|
|
|
searches constructor types in the order in which they appear
|
|
|
|
in the DDS, but there is no requirement that it do so.
|
|
|
|
|
|
|
|
Note: If a dataset contains two constructor types which have field names
|
|
|
|
that are the same (say point.x and pair.x) one should use fully qualified
|
|
|
|
names to get each of those variables.
|
2015-11-30 11:38:12 +08:00
|
|
|
|
2012-08-01 04:34:13 +08:00
|
|
|
*/
|
|
|
|
|
2012-08-03 02:43:21 +08:00
|
|
|
OCerror
|
2012-08-01 04:34:13 +08:00
|
|
|
ocddsdasmerge(OCstate* state, OCnode* dasroot, OCnode* ddsroot)
|
|
|
|
{
|
2012-08-03 02:43:21 +08:00
|
|
|
OCerror stat = OC_NOERR;
|
2017-03-09 08:01:10 +08:00
|
|
|
NClist* dasglobals = nclistnew();
|
|
|
|
NClist* dodsglobals = nclistnew(); /* top-level DODS_XXX {...} */
|
|
|
|
NClist* dasnodes = nclistnew();
|
|
|
|
NClist* varnodes = nclistnew();
|
|
|
|
NClist* ddsnodes;
|
2012-08-01 04:34:13 +08:00
|
|
|
unsigned int i,j;
|
|
|
|
|
|
|
|
if(dasroot->tree == NULL || dasroot->tree->dxdclass != OCDAS)
|
2012-08-03 02:43:21 +08:00
|
|
|
{stat = OCTHROW(OC_EINVAL); goto done;}
|
2012-08-01 04:34:13 +08:00
|
|
|
if(ddsroot->tree == NULL || (ddsroot->tree->dxdclass != OCDDS
|
|
|
|
&& ddsroot->tree->dxdclass != OCDATADDS))
|
2012-08-03 02:43:21 +08:00
|
|
|
{stat = OCTHROW(OC_EINVAL); goto done;}
|
2012-08-01 04:34:13 +08:00
|
|
|
|
|
|
|
ddsnodes = ddsroot->tree->nodes;
|
|
|
|
|
|
|
|
/* 1. collect all the relevant DAS nodes;
|
|
|
|
namely those that contain at least one
|
2015-11-30 11:38:12 +08:00
|
|
|
attribute value, not including the leaf attributes.
|
2012-08-01 04:34:13 +08:00
|
|
|
Simultaneously look for potential ambiguities
|
|
|
|
if found; complain but continue: result are indeterminate.
|
|
|
|
also collect globals separately*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dasroot->tree->nodes);i++) {
|
|
|
|
OCnode* das = (OCnode*)nclistget(dasroot->tree->nodes,i);
|
2012-08-01 04:34:13 +08:00
|
|
|
int hasattributes = 0;
|
|
|
|
if(das->octype == OC_Attribute) continue; /* ignore these for now*/
|
|
|
|
if(das->name == NULL || das->att.isglobal) {
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(dasglobals,(void*)das);
|
2012-08-01 04:34:13 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if(das->att.isdods) {
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(dodsglobals,(void*)das);
|
2012-08-01 04:34:13 +08:00
|
|
|
continue;
|
|
|
|
}
|
2017-03-09 08:01:10 +08:00
|
|
|
for(j=0;j<nclistlength(das->subnodes);j++) {
|
|
|
|
OCnode* subnode = (OCnode*)nclistget(das->subnodes,j);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(subnode->octype == OC_Attribute) {hasattributes = 1; break;}
|
|
|
|
}
|
|
|
|
if(hasattributes) {
|
|
|
|
/* Look for previously collected nodes with same name*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(j=0;j<nclistlength(dasnodes);j++) {
|
|
|
|
OCnode* das2 = (OCnode*)nclistget(dasnodes,j);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(das->name == NULL || das2->name == NULL) continue;
|
|
|
|
if(strcmp(das->name,das2->name)==0) {
|
2017-03-09 08:01:10 +08:00
|
|
|
nclog(NCLOGWARN,"oc_mergedas: potentially ambiguous DAS name: %s",das->name);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
}
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(dasnodes,(void*)das);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 2. collect all the leaf DDS nodes (of type OC_Atomic)*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(ddsnodes);i++) {
|
|
|
|
OCnode* dds = (OCnode*)nclistget(ddsnodes,i);
|
|
|
|
if(dds->octype == OC_Atomic) nclistpush(varnodes,(void*)dds);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* 3. For each das node, locate matching DDS node(s) and attach
|
|
|
|
attributes to the DDS node(s).
|
|
|
|
Match means:
|
|
|
|
1. DAS->fullname :: DDS->fullname
|
2015-11-30 11:38:12 +08:00
|
|
|
2. DAS->name :: DDS->fullname (support DAS names with embedded '.')
|
2012-08-01 04:34:13 +08:00
|
|
|
3. DAS->name :: DDS->name
|
|
|
|
*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dasnodes);i++) {
|
|
|
|
OCnode* das = (OCnode*)nclistget(dasnodes,i);
|
|
|
|
for(j=0;j<nclistlength(varnodes);j++) {
|
|
|
|
OCnode* dds = (OCnode*)nclistget(varnodes,j);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(strcmp(das->fullname,dds->fullname)==0
|
|
|
|
|| strcmp(das->name,dds->fullname)==0
|
|
|
|
|| strcmp(das->name,dds->name)==0) {
|
|
|
|
mergedas1(dds,das);
|
|
|
|
/* remove from dasnodes list*/
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistset(dasnodes,i,(void*)NULL);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-30 11:38:12 +08:00
|
|
|
/* 4. Assign globals*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dasglobals);i++) {
|
|
|
|
OCnode* das = (OCnode*)nclistget(dasglobals,i);
|
2015-11-30 11:38:12 +08:00
|
|
|
if(das == NULL) continue;
|
2012-08-01 04:34:13 +08:00
|
|
|
mergedas1(ddsroot,das);
|
|
|
|
}
|
2015-11-30 11:38:12 +08:00
|
|
|
/* 5. Assign DODS_*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dodsglobals);i++) {
|
|
|
|
OCnode* das = (OCnode*)nclistget(dodsglobals,i);
|
2015-11-30 11:38:12 +08:00
|
|
|
if(das == NULL) continue;
|
2012-08-01 04:34:13 +08:00
|
|
|
mergedods1(ddsroot,das);
|
|
|
|
}
|
2017-03-12 06:20:20 +08:00
|
|
|
|
2015-11-30 11:38:12 +08:00
|
|
|
/* 6. Assign other orphan attributes, which means
|
2017-03-12 07:17:20 +08:00
|
|
|
construct their full name and assign as a global attribute.
|
|
|
|
This is complicated because some servers (e.g. thredds) returns
|
|
|
|
attributes for variables that were not referenced in the DDS.
|
|
|
|
These we continue to suppress.
|
|
|
|
*/
|
|
|
|
mergeother(ddsroot,dasnodes);
|
2012-08-03 02:43:21 +08:00
|
|
|
|
|
|
|
done:
|
2012-08-01 04:34:13 +08:00
|
|
|
/* cleanup*/
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistfree(dasglobals);
|
|
|
|
nclistfree(dodsglobals);
|
|
|
|
nclistfree(dasnodes);
|
|
|
|
nclistfree(varnodes);
|
2012-08-03 02:43:21 +08:00
|
|
|
return OCTHROW(stat);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
|
2015-11-30 11:38:12 +08:00
|
|
|
static OCerror
|
2012-08-01 04:34:13 +08:00
|
|
|
mergedas1(OCnode* dds, OCnode* das)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
2015-11-30 11:38:12 +08:00
|
|
|
OCerror stat = OC_NOERR;
|
2012-08-01 04:34:13 +08:00
|
|
|
if(das == NULL) return OC_NOERR; /* nothing to do */
|
2017-03-09 08:01:10 +08:00
|
|
|
if(dds->attributes == NULL) dds->attributes = nclistnew();
|
2012-08-01 04:34:13 +08:00
|
|
|
/* assign the simple attributes in the das set to this dds node*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(das->subnodes);i++) {
|
|
|
|
OCnode* attnode = (OCnode*)nclistget(das->subnodes,i);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(attnode->octype == OC_Attribute) {
|
2017-11-23 05:10:01 +08:00
|
|
|
OCattribute* att;
|
2017-03-12 07:17:20 +08:00
|
|
|
if(dds->octype == OC_Atomic
|
|
|
|
|| dds->octype == OC_Sequence
|
|
|
|
|| dds->octype == OC_Structure
|
|
|
|
|| dds->octype == OC_Grid)
|
|
|
|
attnode->att.var = dds;
|
2017-11-23 05:10:01 +08:00
|
|
|
att = makeattribute(attnode->name,
|
|
|
|
attnode->etype,
|
|
|
|
attnode->att.values);
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(dds->attributes,(void*)att);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return OCTHROW(stat);
|
|
|
|
}
|
|
|
|
|
2015-11-30 11:38:12 +08:00
|
|
|
static OCerror
|
2012-08-01 04:34:13 +08:00
|
|
|
mergedods1(OCnode* dds, OCnode* dods)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
2015-11-30 11:38:12 +08:00
|
|
|
OCerror stat = OC_NOERR;
|
2012-08-01 04:34:13 +08:00
|
|
|
if(dods == NULL) return OC_NOERR; /* nothing to do */
|
|
|
|
OCASSERT(dods->octype == OC_Attributeset);
|
2017-03-09 08:01:10 +08:00
|
|
|
if(dds->attributes == NULL) dds->attributes = nclistnew();
|
2012-08-01 04:34:13 +08:00
|
|
|
/* assign the simple attributes in the das set to this dds node
|
|
|
|
with renaming to tag as DODS_
|
|
|
|
*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dods->subnodes);i++) {
|
|
|
|
OCnode* attnode = (OCnode*)nclistget(dods->subnodes,i);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(attnode->octype == OC_Attribute) {
|
|
|
|
OCattribute* att;
|
|
|
|
/* prefix the attribute name with the name of the attribute
|
|
|
|
set plus "."
|
|
|
|
*/
|
|
|
|
size_t len = strlen(attnode->name)
|
|
|
|
+ strlen(dods->name)
|
2017-11-24 01:55:24 +08:00
|
|
|
+ strlen(".");
|
|
|
|
len++; /*strlcat nul*/
|
|
|
|
char* newname = (char*)malloc(len+1);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(newname == NULL) return OC_ENOMEM;
|
2017-11-24 01:55:24 +08:00
|
|
|
strncpy(newname,dods->name,len);
|
|
|
|
strlcat(newname,".",len);
|
|
|
|
strlcat(newname,attnode->name,len);
|
2015-11-30 11:38:12 +08:00
|
|
|
att = makeattribute(newname,attnode->etype,attnode->att.values);
|
2012-08-01 04:34:13 +08:00
|
|
|
free(newname);
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(dds->attributes,(void*)att);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return OCTHROW(stat);
|
|
|
|
}
|
|
|
|
|
2017-03-12 07:17:20 +08:00
|
|
|
static OCerror
|
2017-03-18 07:20:02 +08:00
|
|
|
mergeother(OCnode* ddsroot, NClist* dasnodes)
|
2017-03-12 07:17:20 +08:00
|
|
|
{
|
|
|
|
OCerror stat = OC_NOERR;
|
|
|
|
int i;
|
2017-03-18 07:20:02 +08:00
|
|
|
for(i=0;i<nclistlength(dasnodes);i++) {
|
|
|
|
OCnode* das = (OCnode*)nclistget(dasnodes,i);
|
2017-03-12 07:17:20 +08:00
|
|
|
if(das == NULL) continue;
|
|
|
|
if((stat = mergeother1(ddsroot, das))) break;
|
|
|
|
}
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
|
2015-11-30 11:38:12 +08:00
|
|
|
static OCerror
|
|
|
|
mergeother1(OCnode* root, OCnode* das)
|
2012-08-01 04:34:13 +08:00
|
|
|
{
|
2015-11-30 11:38:12 +08:00
|
|
|
OCerror stat = OC_NOERR;
|
|
|
|
OCattribute* att = NULL;
|
|
|
|
|
|
|
|
OCASSERT(root != NULL);
|
2017-03-09 08:01:10 +08:00
|
|
|
if(root->attributes == NULL) root->attributes = nclistnew();
|
2015-11-30 11:38:12 +08:00
|
|
|
|
2017-03-12 07:17:20 +08:00
|
|
|
/* Only include if this is not connected to a variable */
|
|
|
|
if(das->att.var != NULL) goto done;
|
|
|
|
|
2015-11-30 11:38:12 +08:00
|
|
|
if(das->octype == OC_Attribute) {
|
|
|
|
/* compute the full name of this attribute */
|
|
|
|
computefullname(das);
|
|
|
|
/* create attribute */
|
|
|
|
att = makeattribute(das->fullname,das->etype,das->att.values);
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(root->attributes,(void*)att);
|
2015-11-30 11:38:12 +08:00
|
|
|
} else if(das->octype == OC_Attributeset) {
|
|
|
|
int i;
|
|
|
|
/* Recurse */
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(das->subnodes);i++) {
|
|
|
|
OCnode* sub = (OCnode*)nclistget(das->subnodes,i);
|
2015-11-30 11:38:12 +08:00
|
|
|
if(sub == NULL) continue;
|
|
|
|
mergeother1(root,sub);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
stat = OC_EDAS;
|
2017-03-12 07:17:20 +08:00
|
|
|
done:
|
2012-08-01 04:34:13 +08:00
|
|
|
return OCTHROW(stat);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
ocuncorrelate(OCnode* root)
|
|
|
|
{
|
|
|
|
OCtree* tree = root->tree;
|
|
|
|
unsigned int i;
|
|
|
|
if(tree == NULL) return;
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(tree->nodes);i++) {
|
|
|
|
OCnode* node = (OCnode*)nclistget(tree->nodes,i);
|
2012-08-01 04:34:13 +08:00
|
|
|
node->datadds = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static OCerror
|
|
|
|
occorrelater(OCnode* dds, OCnode* dxd)
|
|
|
|
{
|
|
|
|
int i,j;
|
|
|
|
OCerror ocstat = OC_NOERR;
|
|
|
|
|
|
|
|
if(dds->octype != dxd->octype) {
|
|
|
|
OCTHROWCHK((ocstat = OC_EINVAL)); goto fail;
|
|
|
|
}
|
|
|
|
if(dxd->name != NULL && dxd->name != NULL
|
|
|
|
&& strcmp(dxd->name,dds->name) != 0) {
|
|
|
|
OCTHROWCHK((ocstat = OC_EINVAL)); goto fail;
|
2017-12-01 00:22:14 +08:00
|
|
|
} else if(dxd->name != dds->name) { /* test NULL==NULL */
|
|
|
|
OCTHROWCHK((ocstat = OC_EINVAL)); goto fail;
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if(dxd->array.rank != dds->array.rank) {
|
|
|
|
OCTHROWCHK((ocstat = OC_EINVAL)); goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
dds->datadds = dxd;
|
|
|
|
|
|
|
|
switch (dds->octype) {
|
|
|
|
case OC_Dataset:
|
|
|
|
case OC_Structure:
|
|
|
|
case OC_Grid:
|
|
|
|
case OC_Sequence:
|
|
|
|
/* Remember: there may be fewer datadds fields than dds fields */
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dxd->subnodes);i++) {
|
|
|
|
OCnode* dxd1 = (OCnode*)nclistget(dxd->subnodes,(size_t)i);
|
|
|
|
for(j=0;j<nclistlength(dds->subnodes);j++) {
|
|
|
|
OCnode* dds1 = (OCnode*)nclistget(dds->subnodes,(size_t)j);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(strcmp(dxd1->name,dds1->name) == 0) {
|
|
|
|
ocstat = occorrelater(dds1,dxd1);
|
|
|
|
if(ocstat != OC_NOERR) {OCTHROWCHK(ocstat); goto fail;}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case OC_Dimension:
|
|
|
|
case OC_Atomic:
|
|
|
|
break;
|
|
|
|
default: OCPANIC1("unexpected node type: %d",dds->octype);
|
|
|
|
}
|
|
|
|
/* Correlate the dimensions */
|
|
|
|
if(dds->array.rank > 0) {
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dxd->subnodes);i++) {
|
|
|
|
OCnode* ddsdim = (OCnode*)nclistget(dds->array.dimensions,(size_t)i);
|
|
|
|
OCnode* dxddim = (OCnode*)nclistget(dxd->array.dimensions,(size_t)i);
|
2012-08-01 04:34:13 +08:00
|
|
|
ocstat = occorrelater(ddsdim,dxddim);
|
|
|
|
if(!ocstat) goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fail:
|
|
|
|
return OCTHROW(ocstat);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
OCerror
|
|
|
|
occorrelate(OCnode* dds, OCnode* dxd)
|
|
|
|
{
|
|
|
|
if(dds == NULL || dxd == NULL) return OCTHROW(OC_EINVAL);
|
|
|
|
ocuncorrelate(dds);
|
|
|
|
return occorrelater(dds,dxd);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Mark cacheable those atomic String/URL typed nodes
|
|
|
|
that are contained only in structures with rank > 0.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
ocmarkcacheable(OCstate* state, OCnode* ddsroot)
|
|
|
|
{
|
|
|
|
int i,j;
|
|
|
|
#if 0
|
|
|
|
int ok;
|
|
|
|
#endif
|
2017-03-09 08:01:10 +08:00
|
|
|
NClist* treenodes = ddsroot->tree->nodes;
|
|
|
|
NClist* path = nclistnew();
|
|
|
|
for(i=0;i<nclistlength(treenodes);i++) {
|
|
|
|
OCnode* node = (OCnode*)nclistget(treenodes,(size_t)i);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(node->octype != OC_Atomic) continue;
|
|
|
|
if(node->etype != OC_String && node->etype != OC_URL) continue;
|
|
|
|
/* collect node path */
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistclear(path);
|
2012-08-01 04:34:13 +08:00
|
|
|
occollectpathtonode(node,path);
|
|
|
|
#if 0
|
|
|
|
ok = 1;
|
|
|
|
#endif
|
2017-03-09 08:01:10 +08:00
|
|
|
for(j=1;j<nclistlength(path)-1;j++) {/* skip top level dataset and node itself*/
|
|
|
|
OCnode* pathnode = (OCnode*)nclistget(path,(size_t)j);
|
2012-08-01 04:34:13 +08:00
|
|
|
if(pathnode->octype != OC_Structure
|
|
|
|
|| pathnode->array.rank > 0) {
|
|
|
|
#if 0
|
|
|
|
ok=0;
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#if 0
|
|
|
|
if(ok) {
|
|
|
|
node->cache.cacheable = 1;
|
|
|
|
node->cache.valid = 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistfree(path);
|
2012-08-01 04:34:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
2015-11-30 11:38:12 +08:00
|
|
|
|
|
|
|
OCerror
|
|
|
|
ocddsdasmerge(OCstate* state, OCnode* ddsroot, OCnode* dasroot)
|
|
|
|
{
|
|
|
|
int i,j;
|
|
|
|
OCerror stat = OC_NOERR;
|
2017-03-09 08:01:10 +08:00
|
|
|
NClist* globals = nclistnew();
|
2015-11-30 11:38:12 +08:00
|
|
|
if(dasroot == NULL) return OCTHROW(stat);
|
|
|
|
/* Start by looking for global attributes*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dasroot->subnodes);i++) {
|
|
|
|
OCnode* node = (OCnode*)nclistget(dasroot->subnodes,i);
|
2015-11-30 11:38:12 +08:00
|
|
|
if(node->att.isglobal) {
|
2017-03-09 08:01:10 +08:00
|
|
|
for(j=0;j<nclistlength(node->subnodes);j++) {
|
|
|
|
OCnode* attnode = (OCnode*)nclistget(node->subnodes,j);
|
2015-11-30 11:38:12 +08:00
|
|
|
Attribute* att = makeattribute(attnode->name,
|
|
|
|
attnode->etype,
|
|
|
|
attnode->att.values);
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(globals,(void*)att);
|
2015-11-30 11:38:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ddsroot->attributes = globals;
|
|
|
|
/* Now try to match subnode names with attribute set names*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(dasroot->subnodes);i++) {
|
|
|
|
OCnode* das = (OCnode*)nclistget(dasroot->subnodes,i);
|
2015-11-30 11:38:12 +08:00
|
|
|
int match = 0;
|
|
|
|
if(das->att.isglobal) continue;
|
|
|
|
if(das->octype == OC_Attributeset) {
|
2017-03-09 08:01:10 +08:00
|
|
|
for(j=0;j<nclistlength(ddsroot->subnodes) && !match;j++) {
|
|
|
|
OCnode* dds = (OCnode*)nclistget(ddsroot->subnodes,j);
|
2015-11-30 11:38:12 +08:00
|
|
|
if(strcmp(das->name,dds->name) == 0) {
|
|
|
|
match = 1;
|
|
|
|
stat = mergedas1(dds,das);
|
|
|
|
if(stat != OC_NOERR) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!match) {marklostattribute(das);}
|
|
|
|
}
|
|
|
|
if(stat == OC_NOERR) ddsroot->attributed = 1;
|
|
|
|
return OCTHROW(stat);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Merge das attributes into the dds node*/
|
|
|
|
|
|
|
|
static int
|
|
|
|
mergedas1(OCnode* dds, OCnode* das)
|
|
|
|
{
|
|
|
|
int i,j;
|
|
|
|
int stat = OC_NOERR;
|
2017-03-09 08:01:10 +08:00
|
|
|
if(dds->attributes == NULL) dds->attributes = nclistnew();
|
2015-11-30 11:38:12 +08:00
|
|
|
/* assign the simple attributes in the das set to this dds node*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(das->subnodes);i++) {
|
|
|
|
OCnode* attnode = (OCnode*)nclistget(das->subnodes,i);
|
2015-11-30 11:38:12 +08:00
|
|
|
if(attnode->octype == OC_Attribute) {
|
|
|
|
Attribute* att = makeattribute(attnode->name,
|
|
|
|
attnode->etype,
|
|
|
|
attnode->att.values);
|
2017-03-09 08:01:10 +08:00
|
|
|
nclistpush(dds->attributes,(void*)att);
|
2015-11-30 11:38:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Try to merge any enclosed sets with subnodes of dds*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(i=0;i<nclistlength(das->subnodes);i++) {
|
|
|
|
OCnode* dasnode = (OCnode*)nclistget(das->subnodes,i);
|
2015-11-30 11:38:12 +08:00
|
|
|
int match = 0;
|
|
|
|
if(dasnode->octype == OC_Attribute) continue; /* already dealt with above*/
|
2017-03-09 08:01:10 +08:00
|
|
|
for(j=0;j<nclistlength(dds->subnodes) && !match;j++) {
|
|
|
|
OCnode* ddsnode = (OCnode*)nclistget(dds->subnodes,j);
|
2015-11-30 11:38:12 +08:00
|
|
|
if(strcmp(dasnode->name,ddsnode->name) == 0) {
|
|
|
|
match = 1;
|
|
|
|
stat = mergedas1(ddsnode,dasnode);
|
|
|
|
if(stat != OC_NOERR) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!match) {marklostattribute(dasnode);}
|
|
|
|
}
|
|
|
|
return OCTHROW(stat);
|
|
|
|
}
|
|
|
|
|
2012-08-01 04:34:13 +08:00
|
|
|
void*
|
|
|
|
oclinearize(OCtype etype, unsigned int nstrings, char** strings)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
size_t typesize;
|
|
|
|
char* memp;
|
|
|
|
char* memory;
|
|
|
|
|
|
|
|
if(nstrings == 0) return NULL;
|
|
|
|
typesize = octypesize(etype);
|
|
|
|
memory = (char*)ocmalloc(nstrings*typesize);
|
|
|
|
MEMCHECK(memory,NULL);
|
|
|
|
memp = memory;
|
|
|
|
for(i=0;i<nstrings;i++) {
|
|
|
|
char* value = strings[i];
|
|
|
|
converttype(etype,value,memp);
|
|
|
|
memp += typesize;
|
|
|
|
}
|
|
|
|
return memory;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
converttype(OCtype etype, char* value, char* memory)
|
|
|
|
{
|
|
|
|
long iv;
|
|
|
|
unsigned long uiv;
|
|
|
|
double dv;
|
|
|
|
char c[1];
|
|
|
|
int outofrange = 0;
|
|
|
|
#ifdef HAVE_LONG_LONG_INT
|
|
|
|
long long llv;
|
|
|
|
unsigned long long ullv;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
switch (etype) {
|
|
|
|
case OC_Char:
|
|
|
|
if(sscanf(value,"%c",c) != 1) goto fail;
|
|
|
|
*((char*)memory) = c[0];
|
|
|
|
break;
|
|
|
|
case OC_Byte:
|
|
|
|
if(sscanf(value,"%ld",&iv) != 1) goto fail;
|
|
|
|
else if(iv > OC_BYTE_MAX || iv < OC_BYTE_MIN) {iv = OC_BYTE_MAX; outofrange = 1;}
|
|
|
|
*((signed char*)memory) = (signed char)iv;
|
|
|
|
break;
|
|
|
|
case OC_UByte:
|
|
|
|
if(sscanf(value,"%lu",&uiv) != 1) goto fail;
|
|
|
|
else if(uiv > OC_UBYTE_MAX) {uiv = OC_UBYTE_MAX; outofrange = 1;}
|
|
|
|
*((unsigned char*)memory) = (unsigned char)uiv;
|
|
|
|
break;
|
|
|
|
case OC_Int16:
|
|
|
|
if(sscanf(value,"%ld",&iv) != 1) goto fail;
|
|
|
|
else if(iv > OC_INT16_MAX || iv < OC_INT16_MIN) {iv = OC_INT16_MAX; outofrange = 1;}
|
|
|
|
*((signed short*)memory) = (signed short)iv;
|
|
|
|
break;
|
|
|
|
case OC_UInt16:
|
|
|
|
if(sscanf(value,"%lu",&uiv) != 1) goto fail;
|
|
|
|
else if(uiv > OC_UINT16_MAX) {uiv = OC_UINT16_MAX; outofrange = 1;}
|
|
|
|
*((unsigned short*)memory) = (unsigned short)uiv;
|
|
|
|
break;
|
|
|
|
case OC_Int32:
|
|
|
|
if(sscanf(value,"%ld",&iv) != 1) goto fail;
|
|
|
|
else if(iv > OC_INT32_MAX || iv < OC_INT32_MIN) {iv = OC_INT32_MAX; outofrange = 1;}
|
|
|
|
*((signed int*)memory) = (signed int)iv;
|
|
|
|
break;
|
|
|
|
case OC_UInt32:
|
|
|
|
if(sscanf(value,"%lu",&uiv) != 1) goto fail;
|
|
|
|
else if(uiv > OC_UINT32_MAX) {uiv = OC_UINT32_MAX; outofrange = 1;}
|
|
|
|
*((unsigned char*)memory) = (unsigned int)uiv;
|
|
|
|
break;
|
|
|
|
#ifdef HAVE_LONG_LONG_INT
|
|
|
|
case OC_Int64:
|
|
|
|
if(sscanf(value,"%lld",&llv) != 1) goto fail;
|
|
|
|
/*else if(iv > OC_INT64_MAX || iv < OC_INT64_MIN) goto fail;*/
|
|
|
|
*((signed long long*)memory) = (signed long long)llv;
|
|
|
|
break;
|
|
|
|
case OC_UInt64:
|
|
|
|
if(sscanf(value,"%llu",&ullv) != 1) goto fail;
|
|
|
|
*((unsigned long long*)memory) = (unsigned long long)ullv;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case OC_Float32:
|
|
|
|
if(sscanf(value,"%lf",&dv) != 1) goto fail;
|
|
|
|
*((float*)memory) = (float)dv;
|
|
|
|
break;
|
|
|
|
case OC_Float64:
|
|
|
|
if(sscanf(value,"%lf",&dv) != 1) goto fail;
|
|
|
|
*((double*)memory) = (double)dv;
|
|
|
|
break;
|
|
|
|
case OC_String: case OC_URL:
|
|
|
|
*((char**)memory) = nulldup(value);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
if(outofrange)
|
|
|
|
oc_log(LOGWARN,"converttype range failure: %d: %s",etype,value);
|
|
|
|
return 1;
|
|
|
|
fail:
|
|
|
|
oc_log(LOGERR,"converttype bad value: %d: %s",etype,value);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif /*0*/
|