diff options
-rw-r--r-- | src/ucl/src/ucl_internal.h | 2 | ||||
-rw-r--r-- | src/ucl/src/ucl_parser.c | 1 | ||||
-rw-r--r-- | src/ucl/src/ucl_util.c | 63 |
3 files changed, 47 insertions, 19 deletions
diff --git a/src/ucl/src/ucl_internal.h b/src/ucl/src/ucl_internal.h index 8db697736..e2a6d52fa 100644 --- a/src/ucl/src/ucl_internal.h +++ b/src/ucl/src/ucl_internal.h @@ -159,6 +159,8 @@ size_t ucl_unescape_json_string (char *str, size_t len); */ bool ucl_include_handler (const unsigned char *data, size_t len, void* ud); +bool ucl_try_include_handler (const unsigned char *data, size_t len, void* ud); + /** * Handle includes macro * @param data include data diff --git a/src/ucl/src/ucl_parser.c b/src/ucl/src/ucl_parser.c index f2e5001e6..f18a9b671 100644 --- a/src/ucl/src/ucl_parser.c +++ b/src/ucl/src/ucl_parser.c @@ -1753,6 +1753,7 @@ ucl_parser_new (int flags) memset (new, 0, sizeof (struct ucl_parser)); ucl_parser_register_macro (new, "include", ucl_include_handler, new); + ucl_parser_register_macro (new, "try_include", ucl_try_include_handler, new); ucl_parser_register_macro (new, "includes", ucl_includes_handler, new); new->flags = flags; diff --git a/src/ucl/src/ucl_util.c b/src/ucl/src/ucl_util.c index ba6e08c1c..7dad6f32d 100644 --- a/src/ucl/src/ucl_util.c +++ b/src/ucl/src/ucl_util.c @@ -339,7 +339,8 @@ ucl_curl_write_callback (void* contents, size_t size, size_t nmemb, void* ud) * @return */ static bool -ucl_fetch_url (const unsigned char *url, unsigned char **buf, size_t *buflen, UT_string **err) +ucl_fetch_url (const unsigned char *url, unsigned char **buf, size_t *buflen, + UT_string **err, bool must_exist) { #ifdef HAVE_FETCH_H @@ -354,8 +355,10 @@ ucl_fetch_url (const unsigned char *url, unsigned char **buf, size_t *buflen, UT return false; } if ((in = fetchXGet (fetch_url, &us, "")) == NULL) { - ucl_create_err (err, "cannot fetch URL %s: %s", + if (!must_exist) { + ucl_create_err (err, "cannot fetch URL %s: %s", url, strerror (errno)); + } fetchFreeURL (fetch_url); return false; } @@ -402,8 +405,10 @@ ucl_fetch_url (const unsigned char *url, unsigned char **buf, size_t *buflen, UT curl_easy_setopt (curl, CURLOPT_WRITEDATA, &cbdata); if ((r = curl_easy_perform (curl)) != CURLE_OK) { - ucl_create_err (err, "error fetching URL %s: %s", + if (!must_exist) { + ucl_create_err (err, "error fetching URL %s: %s", url, curl_easy_strerror (r)); + } curl_easy_cleanup (curl); if (cbdata.buf) { free (cbdata.buf); @@ -429,14 +434,17 @@ ucl_fetch_url (const unsigned char *url, unsigned char **buf, size_t *buflen, UT * @return */ static bool -ucl_fetch_file (const unsigned char *filename, unsigned char **buf, size_t *buflen, UT_string **err) +ucl_fetch_file (const unsigned char *filename, unsigned char **buf, size_t *buflen, + UT_string **err, bool must_exist) { int fd; struct stat st; if (stat (filename, &st) == -1 || !S_ISREG (st.st_mode)) { - ucl_create_err (err, "cannot stat file %s: %s", - filename, strerror (errno)); + if (must_exist) { + ucl_create_err (err, "cannot stat file %s: %s", + filename, strerror (errno)); + } return false; } if (st.st_size == 0) { @@ -522,7 +530,7 @@ ucl_sig_check (const unsigned char *data, size_t datalen, */ static bool ucl_include_url (const unsigned char *data, size_t len, - struct ucl_parser *parser, bool check_signature) + struct ucl_parser *parser, bool check_signature, bool must_exist) { bool res; @@ -534,8 +542,8 @@ ucl_include_url (const unsigned char *data, size_t len, snprintf (urlbuf, sizeof (urlbuf), "%.*s", (int)len, data); - if (!ucl_fetch_url (urlbuf, &buf, &buflen, &parser->err)) { - return false; + if (!ucl_fetch_url (urlbuf, &buf, &buflen, &parser->err, must_exist)) { + return (!must_exist || false); } if (check_signature) { @@ -544,7 +552,7 @@ ucl_include_url (const unsigned char *data, size_t len, size_t siglen = 0; /* We need to check signature first */ snprintf (urlbuf, sizeof (urlbuf), "%.*s.sig", (int)len, data); - if (!ucl_fetch_file (urlbuf, &sigbuf, &siglen, &parser->err)) { + if (!ucl_fetch_url (urlbuf, &sigbuf, &siglen, &parser->err, true)) { return false; } if (!ucl_sig_check (buf, buflen, sigbuf, siglen, parser)) { @@ -591,7 +599,7 @@ ucl_include_url (const unsigned char *data, size_t len, */ static bool ucl_include_file (const unsigned char *data, size_t len, - struct ucl_parser *parser, bool check_signature) + struct ucl_parser *parser, bool check_signature, bool must_exist) { bool res; struct ucl_chunk *chunk; @@ -602,14 +610,17 @@ ucl_include_file (const unsigned char *data, size_t len, snprintf (filebuf, sizeof (filebuf), "%.*s", (int)len, data); if (realpath (filebuf, realbuf) == NULL) { + if (!must_exist) { + return true; + } ucl_create_err (&parser->err, "cannot open file %s: %s", filebuf, strerror (errno)); return false; } - if (!ucl_fetch_file (realbuf, &buf, &buflen, &parser->err)) { - return false; + if (!ucl_fetch_file (realbuf, &buf, &buflen, &parser->err, must_exist)) { + return (!must_exist || false); } if (check_signature) { @@ -618,7 +629,7 @@ ucl_include_file (const unsigned char *data, size_t len, size_t siglen = 0; /* We need to check signature first */ snprintf (filebuf, sizeof (filebuf), "%s.sig", realbuf); - if (!ucl_fetch_file (filebuf, &sigbuf, &siglen, &parser->err)) { + if (!ucl_fetch_file (filebuf, &sigbuf, &siglen, &parser->err, true)) { return false; } if (!ucl_sig_check (buf, buflen, sigbuf, siglen, parser)) { @@ -675,10 +686,10 @@ ucl_include_handler (const unsigned char *data, size_t len, void* ud) if (*data == '/' || *data == '.') { /* Try to load a file */ - return ucl_include_file (data, len, parser, false); + return ucl_include_file (data, len, parser, false, true); } - return ucl_include_url (data, len, parser, false); + return ucl_include_url (data, len, parser, false, true); } /** @@ -696,10 +707,24 @@ ucl_includes_handler (const unsigned char *data, size_t len, void* ud) if (*data == '/' || *data == '.') { /* Try to load a file */ - return ucl_include_file (data, len, parser, true); + return ucl_include_file (data, len, parser, true, true); + } + + return ucl_include_url (data, len, parser, true, true); +} + + +bool +ucl_try_include_handler (const unsigned char *data, size_t len, void* ud) +{ + struct ucl_parser *parser = ud; + + if (*data == '/' || *data == '.') { + /* Try to load a file */ + return ucl_include_file (data, len, parser, false, false); } - return ucl_include_url (data, len, parser, true); + return ucl_include_url (data, len, parser, false, false); } bool @@ -747,7 +772,7 @@ ucl_parser_add_file (struct ucl_parser *parser, const char *filename) return false; } - if (!ucl_fetch_file (realbuf, &buf, &len, &parser->err)) { + if (!ucl_fetch_file (realbuf, &buf, &len, &parser->err, true)) { return false; } |