Wilcox Development Solutions Blog

Org Mode and Hurl

April 03, 2025

The quest for request makers

As a backend developer I’m often looking to make some one-off requests to test new API endpoints, or to explore some returned data. In the past I’ve used curl directly, or Postman. Postman turning into an enterprise collaboration tool is fine, (or even a cli runner) but in general its gotten to feel too much for simple cases.

I want something simple, reusable, that I can easily commit into Git, that I can pass parameters into easily, and that I can use with other tools I have laying around (let’s be honest: jq).

Meanwhile, my use of Emacs as my day to day driver continues, and I’d like to send requests from it, ideally without demanding that others on my team use the editor ”Exclusively used by Middle Age Computer Scientists”.

In my travels I found two things: hurl and - you guessed it - Org Mode’s Babel package.

NOTE: I’m an org newbie, but some of this stuff isn’t well documented on the web in general, thus my efforts in documenting it here. Excuse any obvious Org mistakes.

Hurl

Hurl is a cross-platform abstraction CLI over CURL. (In fact, my favorite kind of abstraction layer: you can output curl commands from your Hurl configuration!)

Hurl can do more than just sending single requests, it can play a part in your test pipelines, but we’ll focus on the single request cases.

Org Mode Babel

Org Mode is the massively popular, text based, knowledge repository system. Because its built on Emacs it’s also extensible. One of the excellent things you can do with your files in Orgmode is add blocks of executable source code in any language (this is Org Babel). So if you want your document to execute some Ruby and spit out the result, cool, awesome, go for it.

Hurl has an org bable mode in hurl-mode

Hurl primer

Let’s have a quick overview on putting together a request with Hurl. With Hurl you create a file with the hurl request format (quickly documented below) and tell Hurl to execute that request.

Simple Hurl examples

GET https://swapi.dev/api/people/1/
# this is a comment

HTTP/2 200

This is valid HURL and returns us information about everyone’s favorite moisture farmer.

The HURL Request format

The format for making requests in Hurl is sometimes not explicit, but here’s the snippet I use:

GET https://example.com
# GET/POST/ETC URL

# START HEADER SECTION
Content-Type: application/json
# END HEADER SECTION

# START SECTIONS: [Options], [Cookies], etc
[Options]
very-verbose: true
variable: a-variable=some-value
# END SECTIONS

# START BODY
{
}
# END BODY
# RESPOSE CODE ASSERTION below! vvv
HTTP 200
# START [Captures]/[Assert] BLOCK

# END  [Captures]/[Assert] BLOCK

There are two interesting parts of this format.

The first is that Hurl allows you to specify variables, which are later referenced. via the {{a-variable}} syntax.

The second is that assertions or capture groups require a response code assertion. It’s easy to trip up here.

Org mode primer

Org Mode is its own markup language, most of which we’ll skip here: org babel lets you put snippets of any supported language into your org mode document and execute it. Let’s see a simple shell command:

#+BEGIN_SRC shell
ls
#+END_SRC

If you execute this in an org mode file you’ll get a directory listing right there in your org document.

Org Babel can do some interesting - and not well documented - stuff.

Output declarations

Maybe you don’t want your command’s output to go to your org document, but to a file instead.

#+BEGIN_SRC shell :results output file :file /tmp/ls.txt
ls
#+END_SRC

the :results output file :file directs the output to the specified file (it’ll even add a little link to said file in your org mode document).

Dependencies

You can have one org babel block depend on another like so:

#+NAME: generatefile
#+BEGIN_SRC shell :results output file :file /tmp/ls.txt
ls
#+END_SRC

#+BEGIN_SRC shell :var forceDependency=generatefile
cat /tmp/ls.txt
#+END_SRC

In the second block the forceDependency variable references the generatefile named block. If you run the second block it will automatically run the first block.

Hurl in org mode

Usin Hurl in Org Mode lets us make use of those tricks above!

Usage of hurl in Org Babel looks like this:

#+BEGIN_SRC hurl
GET https://swapi.dev/api/people/1/

HTTP/2 200
#+END_SRC

Very easy, same hurl syntax as above.

Org Variables Used in Hurl

Hurl itself lets you pass in variables. Let’s say we have two different environments for our API: test.swapi.dev and swapi.dev. We want to make it easy to send requests to whatever one we’re working on at the time.

#+BEGIN_SRC hurl :var hostname="https://swapi.dev"
GET {{hostname}}/api/people/1
#+END_SRC

Org Babel variables are passed into Hurl and referenced via Hurl’s variable syntax.

Neat.

Iin some cases - I ran into this especially when passing in JWT tokens - the way Hurl Mode passes variables to Hurl may be problematic. For those cases I create and assign a Hurl variable in the Options block in the Hurl config.

jq

We can easily integrate JQ into our org mode documents, with jq-mode.

Let’s use Hurl + JQ in our org mode file to return the name of our future moisture farmer.


#+NAME: getPeople
#+BEGIN_SRC hurl :results output file :file /tmp/people_one.json
GET https://swapi.dev/api/people/1/
HTTP/2 200
#+END_SRC

#+RESULTS: getPeople
[[file:/tmp/people_one.json]]

#+NAME: peopleName
#+begin_src jq :in-file /tmp/people_one.json :var forceDependency=getPeople
.name
#+end_src

We have hurl writing the results to a file, then jq mode picking up that file (using dependencies to make sure it’s always generated), then executing a jq command to get the name.

All we need to do is execute the peopleName block: it will execute the hurl command then run the result through jq.

Conclusion

Hurl gives us an excellent abstraction for making HTTP(S) requests on the CLI. Org Babel makes this even better by giving us an excellent Emacs friendly way to execute those commands at arbitrary times: while still being able to share your request library with colleagues.


Tagged with:

Written by Ryan Wilcox Chief Developer, Wilcox Development Solutions... and other things