diff --git a/configure.ac b/configure.ac index d5c2442b4..e0a19a90d 100644 --- a/configure.ac +++ b/configure.ac @@ -523,20 +523,34 @@ AM_PROG_CC_C_O AC_C_CONST # CURLOPT_KEYPASSWD is not defined until curl version 7.16.4 +# CURLOPT_RESPONSE_CODE is not defined until curl version 7.10.7 # Save/restore CFLAGS SAVECFLAGS="$CFLAGS" CFLAGS="${curl_cflags}" + AC_COMPILE_IFELSE([AC_LANG_PROGRAM( [#include "curl/curl.h"], [[int x = CURLOPT_KEYPASSWD;]])], [havekeypassword=yes], [havekeypassword=no]) -AC_MSG_CHECKING([whether a CURLOPT_KEYPASSWD is defined]) +AC_MSG_CHECKING([whether CURLOPT_KEYPASSWD is defined]) AC_MSG_RESULT([${havekeypassword}]) if test $havekeypassword = yes; then AC_DEFINE([HAVE_CURLOPT_KEYPASSWD],[1],[Is CURLOPT_KEYPASSWD defined]) fi + +AC_COMPILE_IFELSE([AC_LANG_PROGRAM( +[#include "curl/curl.h"], +[[int x = CURLOPT_RESPONSE_CODE;]])], + [haveresponsecode=yes], + [haveresponsecode=no]) +AC_MSG_CHECKING([whether CURLOPT_RESPONSE_CODE is defined]) +AC_MSG_RESULT([${haveresponsecode}]) +if test $haveresponsecode = yes; then + AC_DEFINE([HAVE_CURLOPT_RESPONSE_CODE],[1],[Is CURLOPT_RESPONSE_CODE defined]) +fi + CFLAGS="$SAVECFLAGS" # Set up libtool. diff --git a/include/netcdf.h b/include/netcdf.h index af7a274fd..5d556bc24 100644 --- a/include/netcdf.h +++ b/include/netcdf.h @@ -368,9 +368,12 @@ by the desired type. */ #define NC_EDAPURL (-74) /**< Malformed DAP URL */ #define NC_EDAPCONSTRAINT (-75) /**< Malformed DAP Constraint*/ #define NC_ETRANSLATION (-76) /**< Untranslatable construct */ +#define NC_EACCESS (-77) /**< Access Failure */ +#define NC_EAUTH (-78) /**< Authorization Failure */ /* Misc. additional errors */ -#define NC_ECANTREMOVE (-77) /**< Can't remove file */ +#define NC_ENOTFOUND (-90) /**< No such file */ +#define NC_ECANTREMOVE (-91) /**< Can't remove file */ /* The following was added in support of netcdf-4. Make all netcdf-4 error codes < -100 so that errors can be added to netcdf-3 if diff --git a/libdap2/cache.c b/libdap2/cache.c index 963595058..2f8b28b47 100644 --- a/libdap2/cache.c +++ b/libdap2/cache.c @@ -218,9 +218,9 @@ buildcachenode34(NCDAPCOMMON* nccomm, if((flags & NCF_PREFETCH_ALL) == 0) ce = buildconstraintstring3(constraint); - ocstat = dap_fetch(nccomm,conn,ce,OCDATADDS,&ocroot); + ncstat = dap_fetch(nccomm,conn,ce,OCDATADDS,&ocroot); nullfree(ce); - if(ocstat) {THROWCHK(ocerrtoncerr(ocstat)); goto done;} + if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto done;} ncstat = buildcdftree34(nccomm,ocroot,OCDATA,&dxdroot); if(ncstat) {THROWCHK(ncstat); goto done;} diff --git a/libdap2/daputil.c b/libdap2/daputil.c index b392a2758..5dd1dc092 100644 --- a/libdap2/daputil.c +++ b/libdap2/daputil.c @@ -697,13 +697,15 @@ deltatime() #endif /* Provide a wrapper for oc_fetch so we can log what it does */ -OCerror +NCerror dap_fetch(NCDAPCOMMON* nccomm, OClink conn, const char* ce, OCdxd dxd, OCddsnode* rootp) { - OCerror ocstat; - char* ext; + NCerror ncstat = NC_NOERR; + OCerror ocstat = OC_NOERR; + char* ext = NULL; OCflags flags = 0; + int httpcode = 0; if(dxd == OCDDS) ext = ".dds"; else if(dxd == OCDAS) ext = ".das"; @@ -747,7 +749,21 @@ dap_fetch(NCDAPCOMMON* nccomm, OClink conn, const char* ce, fprintf(stderr,"fetch: dds:\n"); oc_dumpnode(conn,*rootp); #endif - return ocstat; + + /* Look at the HTTP return code */ + httpcode = oc_httpcode(conn); + if(httpcode < 400) { + ncstat = ocerrtoncerr(ocstat); + } else if(httpcode >= 500) { + ncstat = NC_EDAPSVC; + } else if(httpcode == 401) { + ncstat = NC_EAUTH; + } else if(httpcode == 404) { + ncstat = NC_ENOTFOUND; + } else { + ncstat = NC_EACCESS; + } + return ncstat; } /* Check a name to see if it contains illegal dap characters diff --git a/libdap2/daputil.h b/libdap2/daputil.h index f571b532d..69e82b58d 100644 --- a/libdap2/daputil.h +++ b/libdap2/daputil.h @@ -79,7 +79,7 @@ extern int nc__testurl(const char* parth, char** basename); /* Provide a wrapper for oc_fetch so we can log what it does */ -extern OCerror dap_fetch(struct NCDAPCOMMON*,OClink,const char*,OCdxd,OCobject*); +extern NCerror dap_fetch(struct NCDAPCOMMON*,OClink,const char*,OCdxd,OCobject*); extern int dap_badname(char* name); extern char* dap_repairname(char* name); diff --git a/libdap2/ncdap3a.c b/libdap2/ncdap3a.c index 6933581bb..422316245 100644 --- a/libdap2/ncdap3a.c +++ b/libdap2/ncdap3a.c @@ -223,10 +223,10 @@ fprintf(stderr,"seqcountconstraints: %s\n",ncbytescontents(seqcountconstraints)) /* Fetch the minimal data */ if(FLAGSET(dapcomm->controls,NCF_UNCONSTRAINABLE)) - ocstat = dap_fetch(dapcomm,conn,NULL,OCDATADDS,&ocroot); + ncstat = dap_fetch(dapcomm,conn,NULL,OCDATADDS,&ocroot); else - ocstat = dap_fetch(dapcomm,conn,ncbytescontents(seqcountconstraints),OCDATADDS,&ocroot); - if(ocstat) goto fail; + ncstat = dap_fetch(dapcomm,conn,ncbytescontents(seqcountconstraints),OCDATADDS,&ocroot); + if(ncstat) goto fail; ncstat = buildcdftree34(dapcomm,ocroot,OCDATA,&dxdroot); if(ncstat) goto fail; @@ -614,8 +614,8 @@ fetchtemplatemetadata3(NCDAPCOMMON* dapcomm) ce = nulldup(dapcomm->oc.url->selection); /* Get selection constrained DDS */ - ocstat = dap_fetch(dapcomm,dapcomm->oc.conn,ce,OCDDS,&ocroot); - if(ocstat != OC_NOERR) { + ncstat = dap_fetch(dapcomm,dapcomm->oc.conn,ce,OCDDS,&ocroot); + if(ncstat != NC_NOERR) { /* Special Hack. If the protocol is file, then see if we can get the dds from the .dods file */ @@ -623,21 +623,21 @@ fetchtemplatemetadata3(NCDAPCOMMON* dapcomm) THROWCHK(ocstat); goto done; } /* Fetch the data dds */ - ocstat = dap_fetch(dapcomm,dapcomm->oc.conn,ce,OCDATADDS,&ocroot); - if(ocstat != OC_NOERR) { - THROWCHK(ocstat); goto done; + ncstat = dap_fetch(dapcomm,dapcomm->oc.conn,ce,OCDATADDS,&ocroot); + if(ncstat != NC_NOERR) { + THROWCHK(ncstat); goto done; } /* Note what we did */ nclog(NCLOGWARN,"Cannot locate .dds file, using .dods file"); } /* Get selection constrained DAS */ - ocstat = dap_fetch(dapcomm,dapcomm->oc.conn,ce,OCDAS,&dapcomm->oc.ocdasroot); - if(ocstat != OC_NOERR) { + ncstat = dap_fetch(dapcomm,dapcomm->oc.conn,ce,OCDAS,&dapcomm->oc.ocdasroot); + if(ncstat != NC_NOERR) { /* Ignore but complain */ nclog(NCLOGWARN,"Could not read DAS; ignored"); dapcomm->oc.ocdasroot = NULL; - ocstat = OC_NOERR; + ncstat = NC_NOERR; } /* Construct the netcdf cdf tree corresponding to the dds tree*/ @@ -677,8 +677,8 @@ fetchconstrainedmetadata3(NCDAPCOMMON* dapcomm) else ce = buildconstraintstring3(dapcomm->oc.dapconstraint); { - ocstat = dap_fetch(dapcomm,dapcomm->oc.conn,ce,OCDDS,&ocroot); - if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} + ncstat = dap_fetch(dapcomm,dapcomm->oc.conn,ce,OCDDS,&ocroot); + if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto fail;} /* Construct our parallel dds tree; including attributes*/ ncstat = buildcdftree34(dapcomm,ocroot,OCDDS,&ddsroot); diff --git a/libdispatch/derror.c b/libdispatch/derror.c index c94e55fb1..d1f73673b 100644 --- a/libdispatch/derror.c +++ b/libdispatch/derror.c @@ -189,6 +189,14 @@ nc_strerror(int ncerr1) return "NetCDF: Malformed or unexpected Constraint"; case NC_ETRANSLATION: return "NetCDF: Untranslatable construct"; + case NC_EACCESS: + return "NetCDF: Access failure"; + case NC_EAUTH: + return "NetCDF: Authorization failure"; + case NC_ENOTFOUND: + return "NetCDF: file not found"; + case NC_ECANTREMOVE: + return "NetCDF: cannot delete file"; case NC_EHDFERR: return "NetCDF: HDF error"; case NC_ECANTREAD: diff --git a/oc2/oc.c b/oc2/oc.c index a88064bfe..1ad311cd2 100644 --- a/oc2/oc.c +++ b/oc2/oc.c @@ -1831,6 +1831,23 @@ oc_svcerrordata(OCobject link, char** codep, return OCTHROW(ocsvcerrordata(state,codep,msgp,httpp)); } +/*! +Obtain the HTTP code (e.g. 200, 404, etc) from the last +fetch command. + +\param[in] link The link through which the server is accessed. + +\retval the HTTP code +*/ + +OCerror +oc_httpcode(OCobject link) +{ + OCstate* state; + OCVERIFY(OC_State,link); + OCDEREF(OCstate*,state,link); + return state->error.httpcode; +} /**************************************************/ /* New 10/31/2009: return the size(in bytes) diff --git a/oc2/oc.h b/oc2/oc.h index a29d05c4b..f2aadea9e 100644 --- a/oc2/oc.h +++ b/oc2/oc.h @@ -524,10 +524,16 @@ extern OCerror oc_merge_das(OClink, OCddsnode dasroot, OCddsnode ddsroot); /* Debugging */ /* When a server error is detected, then it is possible - to get the server error info using this procedure */ + to get DODS supplied server error info using this procedure */ extern OCerror oc_svcerrordata(OClink link, char** codep, char** msgp, long* httpp); +/* Get the HTTP return code from the last call; + note that this may or may not be the same as returned + by oc_svcerrordata. + */ +extern int oc_httpcode(OClink); + /**************************************************/ /* Experimental/Undocumented */ diff --git a/oc2/ochttp.c b/oc2/ochttp.c index 7119a2d42..781a46307 100644 --- a/oc2/ochttp.c +++ b/oc2/ochttp.c @@ -26,7 +26,11 @@ ocfetchhttpcode(CURL* curl) long httpcode; CURLcode cstat = CURLE_OK; /* Extract the http code */ +#ifdef HAVE_CURLINFO_RESPONSE_CODE cstat = curl_easy_getinfo(curl,CURLINFO_RESPONSE_CODE,&httpcode); +#else + cstat = curl_easy_getinfo(curl,CURLINFO_HTTP_CODE,&httpcode); +#endif if(cstat != CURLE_OK) httpcode = 0; return httpcode; } @@ -62,6 +66,7 @@ ocfetchurl_file(CURL* curl, const char* url, FILE* stream, fetchdata.stream = stream; fetchdata.size = 0; cstat = curl_easy_perform(curl); + if (cstat != CURLE_OK) goto fail; diff --git a/oc2/ocinternal.c b/oc2/ocinternal.c index 40c2a548f..19f713489 100644 --- a/oc2/ocinternal.c +++ b/oc2/ocinternal.c @@ -267,9 +267,9 @@ ocfetch(OCstate* state, const char* constraint, OCdxd kind, OCflags flags, default: break; }/*switch*/ + /* Obtain any http code */ + state->error.httpcode = ocfetchhttpcode(state->curl); if(stat != OC_NOERR) { - /* Obtain any http code */ - state->error.httpcode = ocfetchhttpcode(state->curl); if(state->error.httpcode >= 400) { oclog(OCLOGWARN,"oc_open: Could not read url; http error = %l",state->error.httpcode); } else {