netcdf-c/libdispatch/nclog.c
Dennis Heimbigner efd905a323 Add tests for filter order on read and write cases
re: https://github.com/Unidata/netcdf-c/issues/1923
re: https://github.com/Unidata/netcdf-c/issues/1921

The issue was raised about the order of returned filter ids
for nc_inq_var_filter_ids() when creating a file as opposed
to later reading the file.

For creation, the order is the same as the order in which the
calls to nc_def_var_filter() occur.
However, after the file is closed and then reopened for reading,
the question was raised if the returned order is the same or the reverse.
In fact the order is the same in both cases.

This PR extends the existing filter order testcase to check the create
versus read orders. This also required changing the H5Znoop(1) filters
in the plugins directory.

Misc. Unrelated Changes
1. fix calls to fdopen under windows
2. Temporarily suppres the nczarr_tests/run_chunkcases test
   since it seems to be causing problems with github actions.
2020-12-29 20:12:35 -07:00

245 lines
5.5 KiB
C

/*********************************************************************
* Copyright 2018, UCAR/Unidata
* See netcdf/COPYRIGHT file for copying and redistribution conditions.
* $Header$
*********************************************************************/
#include "config.h"
#ifdef _MSC_VER
#include<io.h>
#endif
#include <stdlib.h>
#include <stdio.h>
#include <stdarg.h>
#include <string.h>
#ifdef HAVE_FCNTL_H
#include <fcntl.h>
#endif
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#include "nclog.h"
#define PREFIXLEN 8
#define MAXTAGS 256
#define NCTAGDFALT "Log";
static int nclogginginitialized = 0;
static struct NCLOGGLOBAL {
int nclogging;
int ncsystemfile; /* 1 => we are logging to file we did not open */
char* nclogfile;
FILE* nclogstream;
} nclog_global = {0,0,NULL,NULL};
static const char* nctagset[] = {"Note","Warning","Error","Debug"};
static const int nctagsize = sizeof(nctagset)/sizeof(char*);
/* Forward */
static const char* nctagname(int tag);
/*!\defgroup NClog NClog Management
@{*/
/*!\internal
*/
void
ncloginit(void)
{
const char* file;
if(nclogginginitialized)
return;
nclogginginitialized = 1;
memset(&nclog_global,0,sizeof(nclog_global));
ncsetlogging(0);
nclog_global.nclogfile = NULL;
nclog_global.nclogstream = NULL;
/* Use environment variables to preset nclogging state*/
/* I hope this is portable*/
if(getenv(NCENVLOGGING) != NULL) {
ncsetlogging(1);
}
file = getenv(NCENVLOGFILE);
if(file != NULL && strlen(file) > 0) {
nclogopen(file);
}
}
/*!
Enable/Disable logging.
\param[in] tf If 1, then turn on logging, if 0, then turn off logging.
\return The previous value of the logging flag.
*/
int
ncsetlogging(int tf)
{
int was;
if(!nclogginginitialized) ncloginit();
was = nclog_global.nclogging;
nclog_global.nclogging = tf;
return was;
}
/*!
Specify a file into which to place logging output.
\param[in] file The name of the file into which to place logging output.
If the file has the value NULL, then send logging output to
stderr.
\return zero if the open failed, one otherwise.
*/
int
nclogopen(const char* file)
{
if(!nclogginginitialized) ncloginit();
nclogclose();
if(file == NULL || strlen(file) == 0) {
/* use stderr*/
nclog_global.nclogstream = stderr;
nclog_global.nclogfile = NULL;
nclog_global.ncsystemfile = 1;
} else if(strcmp(file,"stdout") == 0) {
/* use stdout*/
nclog_global.nclogstream = stdout;
nclog_global.nclogfile = NULL;
nclog_global.ncsystemfile = 1;
} else if(strcmp(file,"stderr") == 0) {
/* use stderr*/
nclog_global.nclogstream = stderr;
nclog_global.nclogfile = NULL;
nclog_global.ncsystemfile = 1;
} else {
int fd;
nclog_global.nclogfile = strdup(file);
nclog_global.nclogstream = NULL;
/* We need to deal with this file carefully
to avoid unauthorized access*/
fd = open(nclog_global.nclogfile,O_WRONLY|O_APPEND|O_CREAT,0600);
if(fd >= 0) {
nclog_global.nclogstream = fdopen(fd,"a");
} else {
free(nclog_global.nclogfile);
nclog_global.nclogfile = NULL;
nclog_global.nclogstream = NULL;
ncsetlogging(0);
return 0;
}
nclog_global.ncsystemfile = 0;
}
return 1;
}
void
nclogclose(void)
{
if(!nclogginginitialized) ncloginit();
if(nclog_global.nclogstream != NULL && !nclog_global.ncsystemfile) {
fclose(nclog_global.nclogstream);
}
if(nclog_global.nclogfile != NULL) free(nclog_global.nclogfile);
nclog_global.nclogstream = NULL;
nclog_global.nclogfile = NULL;
nclog_global.ncsystemfile = 0;
}
/*!
Send logging messages. This uses a variable
number of arguments and operates like the stdio
printf function.
\param[in] tag Indicate the kind of this log message.
\param[in] format Format specification as with printf.
*/
void
nclog(int tag, const char* fmt, ...)
{
va_list args;
const char* prefix;
int was;
if(!nclogginginitialized) ncloginit();
if(tag == NCLOGERR) was = ncsetlogging(1);
if(!nclog_global.nclogging) goto done;
if(nclog_global.nclogstream == NULL)
nclog_global.nclogstream = stderr;
prefix = nctagname(tag);
fprintf(nclog_global.nclogstream,"%s:",prefix);
if(fmt != NULL) {
va_start(args, fmt);
vfprintf(nclog_global.nclogstream, fmt, args);
va_end( args );
}
fprintf(nclog_global.nclogstream, "\n" );
fflush(nclog_global.nclogstream);
done:
if(tag == NCLOGERR) ncsetlogging(was);
}
void
ncvlog(int tag, const char* fmt, va_list ap)
{
const char* prefix;
if(!nclogginginitialized) ncloginit();
if(!nclog_global.nclogging || nclog_global.nclogstream == NULL) return;
prefix = nctagname(tag);
fprintf(nclog_global.nclogstream,"%s:",prefix);
if(fmt != NULL) {
vfprintf(nclog_global.nclogstream, fmt, ap);
}
fprintf(nclog_global.nclogstream, "\n" );
fflush(nclog_global.nclogstream);
}
void
nclogtext(int tag, const char* text)
{
nclogtextn(tag,text,strlen(text));
}
/*!
Send arbitrarily long text as a logging message.
Each line will be sent using nclog with the specified tag.
\param[in] tag Indicate the kind of this log message.
\param[in] text Arbitrary text to send as a logging message.
*/
void
nclogtextn(int tag, const char* text, size_t count)
{
NC_UNUSED(tag);
if(!nclog_global.nclogging || nclog_global.nclogstream == NULL) return;
fwrite(text,1,count,nclog_global.nclogstream);
fflush(nclog_global.nclogstream);
}
static const char*
nctagname(int tag)
{
if(tag < 0 || tag >= nctagsize)
return "unknown";
return nctagset[tag];
}
/**@}*/