From e9e571b96528e57db76e950e52d4df1acec2c13b Mon Sep 17 00:00:00 2001 From: Daniel Stenberg Date: Tue, 27 Aug 2024 16:22:35 +0200 Subject: [PATCH] trurl: accept "control codes" in get output Like for this: trurl example.com/%18 -g '{path}' It was previously prevented. Added test. Also indented the entire JSON file so check the diff without whitespace if you want to see only this added test. Fixes #338 --- tests.json | 614 +++++++++++++++++++++++++++-------------------------- trurl.c | 19 +- 2 files changed, 332 insertions(+), 301 deletions(-) diff --git a/tests.json b/tests.json index 28f2967e..4cbcff48 100644 --- a/tests.json +++ b/tests.json @@ -2425,225 +2425,225 @@ "returncode": 0 } }, - { - "input": { - "arguments": [ - "0?00%000000000000000000000=0000000000" - ] - }, - "expected": { - "stdout": "http://0.0.0.0/?00%000000000000000000000=0000000000\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "--json", - "0?0%000000000000000000000000000000000", - "--urlencode" - ] - }, - "expected": { - "returncode": 0, - "stderr": "", - "stdout": [ - { - "url": "http://0.0.0.0/?0%000000000000000000000000000000000", - "parts": { - "scheme": "http", - "host": "0.0.0.0", - "path": "/", - "query": "0%000000000000000000000000000000000" - }, - "params": [ - { - "key": "0\u00000000000000000000000000000000000", - "value": "" - } - ] - } - ] - } - }, - { - "input": { - "arguments": [ - "--json", - "0?0%000000000000000000000000000000000=000%0000000000", - "--urlencode" - ] - }, - "expected": { - "returncode": 0, - "stderr": "", - "stdout": [ - { - "url": "http://0.0.0.0/?0%000000000000000000000000000000000=000%0000000000", - "parts": { - "scheme": "http", - "host": "0.0.0.0", - "path": "/", - "query": "0%000000000000000000000000000000000=000%0000000000" - }, - "params": [ - { - "key": "0\u00000000000000000000000000000000000", - "value": "000\u000000000000" - } - ] - } - ] - } - }, - { - "input": { - "arguments": [ - "example.com", - "--set", - "host=[::1]" - ] - }, - "expected": { - "stdout": "http://[::1]/\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "example.com:88", - "--set", - "port?=99" - ] - }, - "expected": { - "stdout": "http://example.com:88/\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "example.com", - "--set", - "port?=99" - ] - }, - "expected": { - "stdout": "http://example.com:99/\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "example.com", - "--append", - "query=add", - "--iterate", - "scheme=http ftp" - ] - }, - "expected": { - "stdout": "http://example.com/?add\nftp://example.com/?add\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "example.com", - "--append", - "path=add", - "--iterate", - "scheme=http ftp" - ] - }, - "expected": { - "stdout": "http://example.com/add\nftp://example.com/add\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "example.com", - "--append", - "path=add", - "--append", - "path=two" - ] - }, - "expected": { - "stdout": "http://example.com/add/two\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "https://curl.se?name=mr%00smith", - "--get", - "{query}" - ] - }, - "expected": { - "stderr": "trurl note: URL decode error, most likely because of rubbish in the input (query)\n", - "returncode": 0, - "stdout": "\n" - } - }, - { - "input": { - "arguments": [ - "https://curl.se?name=mr%00smith", - "--get", - "{strict:query}" - ] - }, - "expected": { - "stderr": "trurl error: problems URL decoding query\ntrurl error: Try trurl -h for help\n", - "returncode": 10, - "stdout": "" - } - }, - { - "required": ["no-guess-scheme"], - "input": { - "arguments": [ - "example.com", - "--set", - "scheme?=https" - ] - }, - "expected": { - "stdout": "https://example.com/\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "ftp://example.com", - "--set", - "scheme?=https" - ] - }, - "expected": { - "stdout": "ftp://example.com/\n", - "stderr": "", - "returncode": 0 - } - }, + { + "input": { + "arguments": [ + "0?00%000000000000000000000=0000000000" + ] + }, + "expected": { + "stdout": "http://0.0.0.0/?00%000000000000000000000=0000000000\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "--json", + "0?0%000000000000000000000000000000000", + "--urlencode" + ] + }, + "expected": { + "returncode": 0, + "stderr": "", + "stdout": [ + { + "url": "http://0.0.0.0/?0%000000000000000000000000000000000", + "parts": { + "scheme": "http", + "host": "0.0.0.0", + "path": "/", + "query": "0%000000000000000000000000000000000" + }, + "params": [ + { + "key": "0\u00000000000000000000000000000000000", + "value": "" + } + ] + } + ] + } + }, + { + "input": { + "arguments": [ + "--json", + "0?0%000000000000000000000000000000000=000%0000000000", + "--urlencode" + ] + }, + "expected": { + "returncode": 0, + "stderr": "", + "stdout": [ + { + "url": "http://0.0.0.0/?0%000000000000000000000000000000000=000%0000000000", + "parts": { + "scheme": "http", + "host": "0.0.0.0", + "path": "/", + "query": "0%000000000000000000000000000000000=000%0000000000" + }, + "params": [ + { + "key": "0\u00000000000000000000000000000000000", + "value": "000\u000000000000" + } + ] + } + ] + } + }, + { + "input": { + "arguments": [ + "example.com", + "--set", + "host=[::1]" + ] + }, + "expected": { + "stdout": "http://[::1]/\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "example.com:88", + "--set", + "port?=99" + ] + }, + "expected": { + "stdout": "http://example.com:88/\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "example.com", + "--set", + "port?=99" + ] + }, + "expected": { + "stdout": "http://example.com:99/\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "example.com", + "--append", + "query=add", + "--iterate", + "scheme=http ftp" + ] + }, + "expected": { + "stdout": "http://example.com/?add\nftp://example.com/?add\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "example.com", + "--append", + "path=add", + "--iterate", + "scheme=http ftp" + ] + }, + "expected": { + "stdout": "http://example.com/add\nftp://example.com/add\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "example.com", + "--append", + "path=add", + "--append", + "path=two" + ] + }, + "expected": { + "stdout": "http://example.com/add/two\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "https://curl.se?name=mr%00smith", + "--get", + "{query}" + ] + }, + "expected": { + "stderr": "trurl note: URL decode error, most likely because of rubbish in the input (query)\n", + "returncode": 0, + "stdout": "\n" + } + }, + { + "input": { + "arguments": [ + "https://curl.se?name=mr%00smith", + "--get", + "{strict:query}" + ] + }, + "expected": { + "stderr": "trurl error: problems URL decoding query\ntrurl error: Try trurl -h for help\n", + "returncode": 10, + "stdout": "" + } + }, + { + "required": ["no-guess-scheme"], + "input": { + "arguments": [ + "example.com", + "--set", + "scheme?=https" + ] + }, + "expected": { + "stdout": "https://example.com/\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "ftp://example.com", + "--set", + "scheme?=https" + ] + }, + "expected": { + "stdout": "ftp://example.com/\n", + "stderr": "", + "returncode": 0 + } + }, { "input": { "arguments": [ @@ -2689,85 +2689,99 @@ "returncode": 0 } }, - { - "input": { - "arguments": [ - "https://example.com/one/t%61o/%2F%42/" - ] - }, - "expected": { - "stdout": "https://example.com/one/tao/%2fB/\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "https://example.com/one/t%61o/%2F%42/", - "--append", - "path=%61" - ] - }, - "expected": { - "stdout": "https://example.com/one/tao/%2fB/%2561\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "https://ex%61mple.com/h%61s/?wh%61t" - ] - }, - "expected": { - "stdout": "https://example.com/has/?what\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "https://example.com/", - "--get", - "{must:query}" - ] - }, - "expected": { - "stdout": "", - "stderr": "trurl error: missing must:query\ntrurl error: Try trurl -h for help\n", - "returncode": 10 - } - }, - { - "input": { - "arguments": [ - "https://example.com/?", - "--get", - "{must:query}" - ] - }, - "required": ["get-empty"], - "expected": { - "stdout": "\n", - "stderr": "", - "returncode": 0 - } - }, - { - "input": { - "arguments": [ - "https://example.com/", - "--get", - "{must:fragment}" - ] - }, - "expected": { - "stdout": "", - "stderr": "trurl error: missing must:fragment\ntrurl error: Try trurl -h for help\n", - "returncode": 10 - } - } + { + "input": { + "arguments": [ + "https://example.com/one/t%61o/%2F%42/" + ] + }, + "expected": { + "stdout": "https://example.com/one/tao/%2fB/\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "https://example.com/one/t%61o/%2F%42/", + "--append", + "path=%61" + ] + }, + "expected": { + "stdout": "https://example.com/one/tao/%2fB/%2561\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "https://ex%61mple.com/h%61s/?wh%61t" + ] + }, + "expected": { + "stdout": "https://example.com/has/?what\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "https://example.com/", + "--get", + "{must:query}" + ] + }, + "expected": { + "stdout": "", + "stderr": "trurl error: missing must:query\ntrurl error: Try trurl -h for help\n", + "returncode": 10 + } + }, + { + "input": { + "arguments": [ + "https://example.com/?", + "--get", + "{must:query}" + ] + }, + "required": ["get-empty"], + "expected": { + "stdout": "\n", + "stderr": "", + "returncode": 0 + } + }, + { + "input": { + "arguments": [ + "https://example.com/", + "--get", + "{must:fragment}" + ] + }, + "expected": { + "stdout": "", + "stderr": "trurl error: missing must:fragment\ntrurl error: Try trurl -h for help\n", + "returncode": 10 + } + }, + { + "input": { + "arguments": [ + "http://example.org/%18", + "--get", + "{path}" + ] + }, + "expected": { + "stdout": "/\u0018\n", + "stderr": "", + "returncode": 0 + } + } ] diff --git a/trurl.c b/trurl.c index dd995c57..c08ee9ba 100644 --- a/trurl.c +++ b/trurl.c @@ -938,7 +938,24 @@ static void get(struct option *o, CURLU *uh) const struct var *v = comp2var(ptr, vlen); if(v) { char *nurl; - CURLUcode rc = geturlpart(o, mods, uh, v->part, &nurl); + /* ask for it URL encode always, to avoid libcurl warning on + content */ + CURLUcode rc = geturlpart(o, mods | VARMODIFIER_URLENCODED, + uh, v->part, &nurl); + if(!rc && !(mods & VARMODIFIER_URLENCODED) && !o->urlencode) { + /* it should not be encoded in the output */ + int olen; + char *dec = curl_easy_unescape(NULL, nurl, 0, &olen); + curl_free(nurl); + if(memchr(dec, '\0', (size_t)olen)) { + /* a binary zero cannot be shown */ + rc = CURLUE_URLDECODE; + curl_free(dec); + dec = NULL; + } + nurl = dec; + } + if(rc == CURLUE_OK) { fputs(nurl, stream); curl_free(nurl);