mirror of
https://github.com/Unidata/netcdf-c.git
synced 2025-01-12 15:45:21 +08:00
f6e25b695e
re: https://github.com/Unidata/netcdf-c/issues/2117 re: https://github.com/Unidata/netcdf-c/issues/2119 * Modify libsrc to allow byte-range reading of netcdf-3 files in private S3 buckets; this required using the aws sdk. Also add a test case. * The aws sdk can sometimes cause problems if the Awd::ShutdownAPI function is not called. So at optional atexit() support to ensure it is called. This is disabled for Windows. * Add documentation to nczarr.md on how to build and use the aws sdk under windows. Currently it builds, but testing fails. * Switch testing from stratus to the Unidata bucket on S3. * Improve support for the s3: url protocol. * Add a s3 specific utility code file: ds3util.c * Modify NC_infermodel to attempt to read the magic number of byte-ranged files in S3. ## Misc. * Move and rename the core S3 SDK wrapper code (libnczarr/zs3sdk.cpp) to libdispatch since it now used in libsrc as well as libnczarr. * Add calls to nc_finalize in the utilities in case atexit is disabled. * Add header only json parser to the distribution rather than as a built source.
325 lines
8.5 KiB
C
325 lines
8.5 KiB
C
/*********************************************************************
|
|
* COPYRIGHT 2018, UCAR/UNIDATA
|
|
* SEE NETCDF/COPYRIGHT FILE FOR COPYING AND REDISTRIBUTION CONDITIONS.
|
|
* ********************************************************************/
|
|
|
|
#if HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#undef DEBUG
|
|
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
#ifdef DEBUG
|
|
#include <stdio.h>
|
|
#endif
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#ifdef HAVE_FCNTL_H
|
|
#include <fcntl.h>
|
|
#endif
|
|
#ifdef _MSC_VER /* MICROSOFT COMPILERS */
|
|
#include <io.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#include "netcdf.h"
|
|
#include "nc3internal.h"
|
|
#include "nclist.h"
|
|
#include "ncbytes.h"
|
|
#include "ncrc.h"
|
|
#include "nclog.h"
|
|
#include "ncio.h"
|
|
#include "fbits.h"
|
|
#include "rnd.h"
|
|
#include "ncs3sdk.h"
|
|
|
|
#define DEFAULTPAGESIZE 16384
|
|
|
|
/* PRIVATE DATA */
|
|
|
|
enum IO {NOIO=0, CURLIO=1, S3IO=2};
|
|
|
|
typedef struct NCS3IO {
|
|
long long size; /* of the object */
|
|
NCS3INFO s3;
|
|
void* s3client;
|
|
char* errmsg;
|
|
void* buffer;
|
|
} NCS3IO;
|
|
|
|
/* Forward */
|
|
static int s3io_rel(ncio *const nciop, off_t offset, int rflags);
|
|
static int s3io_get(ncio *const nciop, off_t offset, size_t extent, int rflags, void **const vpp);
|
|
static int s3io_move(ncio *const nciop, off_t to, off_t from, size_t nbytes, int rflags);
|
|
static int s3io_sync(ncio *const nciop);
|
|
static int s3io_filesize(ncio* nciop, off_t* filesizep);
|
|
static int s3io_pad_length(ncio* nciop, off_t length);
|
|
static int s3io_close(ncio* nciop, int);
|
|
|
|
#define reporterr(s3io) {if((s3io) && (s3io)->errmsg) {nclog(NCLOGERR,(s3io)->errmsg);} nullfree((s3io)->errmsg); (s3io)->errmsg = NULL;}
|
|
|
|
static long pagesize = 0;
|
|
|
|
/* Create a new ncio struct to hold info about the file. */
|
|
static int
|
|
s3io_new(const char* path, int ioflags, ncio** nciopp, NCS3IO** hpp)
|
|
{
|
|
int status = NC_NOERR;
|
|
ncio* nciop = NULL;
|
|
NCS3IO* s3io = NULL;
|
|
|
|
if(pagesize == 0)
|
|
pagesize = DEFAULTPAGESIZE;
|
|
errno = 0;
|
|
|
|
nciop = (ncio* )calloc(1,sizeof(ncio));
|
|
if(nciop == NULL) {status = NC_ENOMEM; goto fail;}
|
|
|
|
nciop->ioflags = ioflags;
|
|
|
|
*((char**)&nciop->path) = strdup(path);
|
|
if(nciop->path == NULL) {status = NC_ENOMEM; goto fail;}
|
|
|
|
*((ncio_relfunc**)&nciop->rel) = s3io_rel;
|
|
*((ncio_getfunc**)&nciop->get) = s3io_get;
|
|
*((ncio_movefunc**)&nciop->move) = s3io_move;
|
|
*((ncio_syncfunc**)&nciop->sync) = s3io_sync;
|
|
*((ncio_filesizefunc**)&nciop->filesize) = s3io_filesize;
|
|
*((ncio_pad_lengthfunc**)&nciop->pad_length) = s3io_pad_length;
|
|
*((ncio_closefunc**)&nciop->close) = s3io_close;
|
|
|
|
s3io = (NCS3IO*)calloc(1,sizeof(NCS3IO));
|
|
if(s3io == NULL) {status = NC_ENOMEM; goto fail;}
|
|
*((void* *)&nciop->pvt) = s3io;
|
|
|
|
if(nciopp) *nciopp = nciop;
|
|
if(hpp) *hpp = s3io;
|
|
|
|
done:
|
|
return status;
|
|
|
|
fail:
|
|
nullfree(s3io);
|
|
if(nciop != NULL) {
|
|
if(nciop->path != NULL) free((char*)nciop->path);
|
|
}
|
|
goto done;
|
|
}
|
|
|
|
/* Create a file, and the ncio struct to go with it. This function is
|
|
only called from nc__create_mp.
|
|
|
|
path - path of file to create.
|
|
ioflags - flags from nc_create
|
|
initialsz - From the netcdf man page: "The argument
|
|
Iinitialsize sets the initial size of the file at creation time."
|
|
igeto -
|
|
igetsz -
|
|
sizehintp - the size of a page of data for buffered reads and writes.
|
|
nciopp - pointer to a pointer that will get location of newly
|
|
created and inited ncio struct.
|
|
mempp - pointer to pointer to the initial memory read.
|
|
*/
|
|
int
|
|
s3io_create(const char* path, int ioflags,
|
|
size_t initialsz,
|
|
off_t igeto, size_t igetsz, size_t* sizehintp,
|
|
void* parameters,
|
|
ncio* *nciopp, void** const mempp)
|
|
{
|
|
return NC_EPERM;
|
|
}
|
|
|
|
/* This function opens the data file. It is only called from nc.c,
|
|
from nc__open_mp and nc_delete_mp.
|
|
|
|
path - path of data file.
|
|
ioflags - flags passed into nc_open.
|
|
igeto - looks like this function can do an initial page get, and
|
|
igeto is going to be the offset for that. But it appears to be
|
|
unused
|
|
igetsz - the size in bytes of initial page get (a.k.a. extent). Not
|
|
ever used in the library.
|
|
sizehintp - the size of a page of data for buffered reads and writes.
|
|
nciopp - pointer to pointer that will get address of newly created
|
|
and inited ncio struct.
|
|
mempp - pointer to pointer to the initial memory read.
|
|
*/
|
|
int
|
|
s3io_open(const char* path,
|
|
int ioflags,
|
|
/* ignored */ off_t igeto, size_t igetsz, size_t* sizehintp,
|
|
/* ignored */ void* parameters,
|
|
ncio* *nciopp,
|
|
/* ignored */ void** const mempp)
|
|
{
|
|
ncio* nciop;
|
|
int status;
|
|
NCS3IO* s3io = NULL;
|
|
size_t sizehint;
|
|
NCURI* url = NULL;
|
|
|
|
if(path == NULL ||* path == 0)
|
|
return EINVAL;
|
|
|
|
/* Create private data */
|
|
if((status = s3io_new(path, ioflags, &nciop, &s3io))) goto done;
|
|
|
|
/* parse path */
|
|
ncuriparse(path,&url);
|
|
if(url == NULL)
|
|
{status = NC_EURL; goto done;}
|
|
|
|
/* Convert to canonical path-style */
|
|
if((status = NC_s3urlprocess(url,&s3io->s3))) goto done;
|
|
/* Verify root path */
|
|
if(s3io->s3.rootkey == NULL)
|
|
{status = NC_EURL; goto done;}
|
|
s3io->s3client = NC_s3sdkcreateclient(&s3io->s3);
|
|
/* Get the size */
|
|
switch (status = NC_s3sdkinfo(s3io->s3client,s3io->s3.bucket,s3io->s3.rootkey,&s3io->size,&s3io->errmsg)) {
|
|
case NC_NOERR: break;
|
|
case NC_EEMPTY:
|
|
s3io->size = 0;
|
|
goto done;
|
|
default:
|
|
goto done;
|
|
}
|
|
|
|
sizehint = pagesize;
|
|
|
|
/* sizehint must be multiple of 8 */
|
|
sizehint = (sizehint / 8) * 8;
|
|
if(sizehint < 8) sizehint = 8;
|
|
|
|
*sizehintp = sizehint;
|
|
*nciopp = nciop;
|
|
done:
|
|
ncurifree(url);
|
|
if(status) {
|
|
reporterr(s3io);
|
|
s3io_close(nciop,0);
|
|
}
|
|
return status;
|
|
}
|
|
|
|
/*
|
|
* Get file size in bytes.
|
|
*/
|
|
static int
|
|
s3io_filesize(ncio* nciop, off_t* filesizep)
|
|
{
|
|
NCS3IO* s3io;
|
|
if(nciop == NULL || nciop->pvt == NULL) return NC_EINVAL;
|
|
s3io = (NCS3IO*)nciop->pvt;
|
|
if(filesizep != NULL) *filesizep = s3io->size;
|
|
return NC_NOERR;
|
|
}
|
|
|
|
/*
|
|
* Sync any changes to disk, then truncate or extend file so its size
|
|
* is length. This is only intended to be called before close, if the
|
|
* file is open for writing and the actual size does not match the
|
|
* calculated size, perhaps as the result of having been previously
|
|
* written in NOFILL mode.
|
|
*/
|
|
static int
|
|
s3io_pad_length(ncio* nciop, off_t length)
|
|
{
|
|
return NC_NOERR; /* do nothing */
|
|
}
|
|
|
|
/* Write out any dirty buffers to disk and
|
|
ensure that next read will get data from disk.
|
|
Sync any changes, then close the open file associated with the ncio
|
|
struct, and free its memory.
|
|
nciop - pointer to ncio to close.
|
|
doUnlink - if true, unlink file
|
|
*/
|
|
|
|
static int
|
|
s3io_close(ncio* nciop, int deleteit)
|
|
{
|
|
int status = NC_NOERR;
|
|
NCS3IO* s3io;
|
|
if(nciop == NULL || nciop->pvt == NULL) return NC_NOERR;
|
|
|
|
s3io = (NCS3IO*)nciop->pvt;
|
|
assert(s3io != NULL);
|
|
|
|
if(s3io->s3client && s3io->s3.bucket && s3io->s3.rootkey) {
|
|
NC_s3sdkclose(s3io->s3client, &s3io->s3, deleteit, &s3io->errmsg);
|
|
}
|
|
s3io->s3client = NULL;
|
|
NC_s3clear(&s3io->s3);
|
|
nullfree(s3io->errmsg);
|
|
nullfree(s3io->buffer);
|
|
nullfree(s3io);
|
|
|
|
if(nciop->path != NULL) free((char*)nciop->path);
|
|
free(nciop);
|
|
return status;
|
|
}
|
|
|
|
/*
|
|
* Request that the region (offset, extent)
|
|
* be made available through *vpp.
|
|
*/
|
|
static int
|
|
s3io_get(ncio* const nciop, off_t offset, size_t extent, int rflags, void** const vpp)
|
|
{
|
|
int status = NC_NOERR;
|
|
NCS3IO* s3io;
|
|
|
|
if(nciop == NULL || nciop->pvt == NULL) {status = NC_EINVAL; goto done;}
|
|
s3io = (NCS3IO*)nciop->pvt;
|
|
|
|
assert(s3io->buffer == NULL);
|
|
if((s3io->buffer = (unsigned char*)malloc(extent))==NULL)
|
|
{status = NC_ENOMEM; goto done;}
|
|
status = NC_s3sdkread(s3io->s3client, s3io->s3.bucket, s3io->s3.rootkey, offset, extent, s3io->buffer, &s3io->errmsg);
|
|
if(status) {reporterr(s3io); goto done;}
|
|
|
|
if(vpp) *vpp = s3io->buffer;
|
|
done:
|
|
return status;
|
|
}
|
|
|
|
/*
|
|
* Like memmove(), safely move possibly overlapping data.
|
|
*/
|
|
static int
|
|
s3io_move(ncio* const nciop, off_t to, off_t from, size_t nbytes, int ignored)
|
|
{
|
|
return NC_EPERM;
|
|
}
|
|
|
|
static int
|
|
s3io_rel(ncio* const nciop, off_t offset, int rflags)
|
|
{
|
|
int status = NC_NOERR;
|
|
NCS3IO* s3io;
|
|
|
|
if(nciop == NULL || nciop->pvt == NULL) {status = NC_EINVAL; goto done;}
|
|
s3io = (NCS3IO*)nciop->pvt;
|
|
nullfree(s3io->buffer);
|
|
s3io->buffer = NULL;
|
|
done:
|
|
return status;
|
|
}
|
|
|
|
/*
|
|
* Write out any dirty buffers to disk and
|
|
* ensure that next read will get data from disk.
|
|
*/
|
|
static int
|
|
s3io_sync(ncio* const nciop)
|
|
{
|
|
return NC_NOERR; /* do nothing */
|
|
}
|
|
|