119

Pretty print curl -i – Rob Allen's DevNotes

 6 years ago
source link: https://akrabat.com/pretty-print-curl-i/
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.

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!

This article was posted on 11 October 2017 in Computing, PHP


About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK