Pretty print curl -i
My favourite tool for working with APIs is curl, though I recognise that lots of people like HTTPie and seem very keen to tell about it every time I mention curl…
With curl, I particularly like using the -i switch to view the status line and headers too without the additional cruft of -v:
This generates an output that looks like this:
$ curl -i https://api.joind.in
HTTP/1.1 200 OK
Date: Wed, 04 Oct 2017 09:51:46 GMT
Server: Apache
X-Powered-By: PHP/5.6.4
Access-Control-Allow-Origin: *
Content-Length: 363
Content-Type: application/json; charset=utf8
{"events":"https:\/\/api.joind.in\/v2.1\/events","hot-events":"https:\/\/api.joind.in\/v2.1\/events?filter=hot","upcoming-events":"https:\/\/api.joind.in\/v2.1\/events?filter=upcoming","past-events":"https:\/\/api.joind.in\/v2.1\/events?filter=past","open-cfps":"https:\/\/api.joind.in\/v2.1\/events?filter=cfp","docs":"http:\/\/joindin.github.io\/joindin-api\/"}
What I would like to do is pretty print the body if it’s JSON or XML so that it’s easier to read. There are some tools out there, like jq which will format JSON, but they can’t seem to cope if the first part of the string is not JSON:
$ curl -i https://api.joind.in | jq
parse error: Invalid numeric literal at line 1, column 9
Solution 1: Use -D flag
The easiest solution is to use the -D flag to send the headers to stderr:
$ curl -s -D "/dev/stderr" https://api.joind.in | jq
HTTP/1.1 200 OK
Date: Wed, 04 Oct 2017 14:56:29 GMT
Server: Apache
X-Powered-By: PHP/5.6.4
Access-Control-Allow-Origin: *
Content-Length: 363
Content-Type: application/json; charset=utf8
{
"events": "https://api.joind.in/v2.1/events",
"hot-events": "https://api.joind.in/v2.1/events?filter=hot",
"upcoming-events": "https://api.joind.in/v2.1/events?filter=upcoming",
"past-events": "https://api.joind.in/v2.1/events?filter=past",
"open-cfps": "https://api.joind.in/v2.1/events?filter=cfp",
"docs": "http://joindin.github.io/joindin-api/"
}
To make this easier, you can add to your ~/.curlrc. Mine looks like this:
$ cat .curlrc
-w "\n"
silent
-D /dev/stderr
Now you never need to think about it.
Solution 2: A simple script to pipe the output through
Another way to do it is via a script that knows where the body starts. We can write this in PHP and it can automatically pretty print XML for us too:
#!/usr/bin/env php
<?php
/**
* Pretty print XML or JSON, when the output includes HTTP headers from curl -i
*/
$input = file_get_contents('php://stdin');
$input = trim($input);
if (substr($input, 0, 5) == 'HTTP/') {
// input is probably the output of curl -i
define('DELIMITER', "\r\n\r\n");
list($headers, $body) = explode(DELIMITER, $input, 2);
if ($body) {
echo $headers . DELIMITER;
$input = trim($body);
}
}
// is it JSON?
if ($data = json_decode($input)) {
echo json_encode($data, JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT);
exit;
}
// is it XML?
libxml_use_internal_errors(true);
if ($simpleXML = simplexml_load_string($input)) {
$dom = dom_import_simplexml($simpleXML)->ownerDocument;
$dom->formatOutput = true;
echo $dom->saveXML();
exit;
}
// it's something else!
echo $input;
Save this as as /usr/local/prettyprint, make it executable with chmod a+w /usr/bin/prettyprint and you’re good to go:
$ curl -i https://api.joind.in | prettyprint
HTTP/1.1 200 OK
Date: Wed, 04 Oct 2017 09:57:07 GMT
Server: Apache
X-Powered-By: PHP/5.6.4
Access-Control-Allow-Origin: *
Content-Length: 363
Content-Type: application/json; charset=utf8
{
"events": "https://api.joind.in/v2.1/events",
"hot-events": "https://api.joind.in/v2.1/events?filter=hot",
"upcoming-events": "https://api.joind.in/v2.1/events?filter=upcoming",
"past-events": "https://api.joind.in/v2.1/events?filter=past",
"open-cfps": "https://api.joind.in/v2.1/events?filter=cfp",
"docs": "http://joindin.github.io/joindin-api/"
}
Pick the solution that works best for you!
You can just use -v verbose to print out the headers and automatically send them to STDERR.
curl -v https://api.joind.in | jq
Sort of the best of both worlds
You can use `curl 'example.com' | json_pp`
Amazing. lol ur suggestions was all i needed!