porcelain
is a package which allows creation of robust,
testable HTTP API’s from R. These can be used to expose any R
calculation to other applications while communicating explicit
expectations for inputs and outputs. It is implemented on top of the
excellent plumber
package and uses plumber
’s
programmatic interface to build its API. The interface, goals and
tradeoffs are very different to plumber, but we think that this leads to
high-quality APIs that are easy to put into production.
The plumber
package makes it trivially easy to create
APIs from R. For example, from the documentation
#* Return the sum of two numbers
#* @param a The first number to add
#* @param b The second number to add
#* @post /sum
function(a, b) {
as.numeric(a) + as.numeric(b)
}
Starting this API is easy,
library(plumber)
# 'plumber.R' is the location of the file shown above
pr("plumber.R") %>%
pr_run(port=8000)
and we can interact with it using curl
:
curl -w"\n" --data "a=4&b=3" "http://localhost:8000/sum"
[7]
This simple example illustrates several issues that
porcelain
tries to solve:
a=x&b=3
will produce)We start with a very different approach to plumber
, one
that is undeniably much more work to get started with as everything will
be implemented in a package. We’ll start by creating a version of the
above example (adding two arguments) which makes explicit some of the
type decisions here.
To begin, we’ll write an endpoint that takes two arguments as query parameters (not body parameters as in the plumber example). We will do this in a small package:
[01;34madd
[0m
├── DESCRIPTION
├── NAMESPACE
├──
[01;34mR
[0m
│ └──
[32mapi.R
[0m
├──
[01;34minst
[0m
│ └──
[01;34mschema
[0m
│ └── numeric.json
└──
[01;34mtests
[0m
├──
[01;34mtestthat
[0m
│ └──
[32mtest-api.R
[0m
└──
[32mtestthat.R
[0m
The DESCRIPTION
file is minimal, importing
porcelain
and suggesting testthat
for the
package tests.
Package: add
Title: Adds Numbers
Version: 1.0.0
Authors@R: c(person("Rich", "FitzJohn", role = c("aut", "cre"),
email = "rich.fitzjohn@gmail.com"))
Description: Adds numbers as an HTTP API.
License: CC0
Encoding: UTF-8
Imports: porcelain
Suggests:
testthat (>= 3.0.0)
Config/testthat/edition: 3
The NAMESPACE
file exports the api
function
(described below)
export(api)
Most of the work is in R/api.R
which implements the API
(there is nothing special about this filename)
add <- function(a, b) {
jsonlite::unbox(a + b)
}
endpoint_add <- function() {
porcelain::porcelain_endpoint$new(
"GET", "/", add,
porcelain::porcelain_input_query(a = "numeric", b = "numeric"),
returning = porcelain::porcelain_returning_json("numeric"))
}
api <- function(validate = FALSE) {
api <- porcelain::porcelain$new(validate = validate)
api$handle(endpoint_add())
api
}
There are three functions here:
add
: this is the function we want the api to runendpoint_add
: this is the porcelain “endpoint” which
formalises the contract between the R function and the HTTP api, we’ll
discuss this in detail belowapi
: this creates the api object (which is itself a
plumber object)With this, you could bring up the api with
add::api()$run()
At this point, the porcelain approach probably just looks like much more work than the 7 lines of plumber code above!
The benefit comes from the type checking we can do in the endpoint function, and the fact that these functions are in a package so are easily testable.
In the endpoint we have
porcelain::porcelain_input_query(a = "numeric", b = "numeric")
which advertises (and enforces) that the query parameters
a
and b
expected and numeric - additional
query parameters provided will raise an error. Similarly, the
argument
returning = porcelain::porcelain_returning_json("numeric")
indicates that we will be returning json with a numeric value. This
is different to the numeric
above, as here it refers to a
json schema - this is included in
the package within inst/schema
as
numeric.json
This is going to turn out to be much more useful with more complex apis.
Finally, the package has tests, and with this we can make explicit some of the questions posed in the motivation section.
test_that("can add two numbers", {
expect_equal(add(1, 2), jsonlite::unbox(3))
expect_equal(add(3.14, 1.23), jsonlite::unbox(4.37))
})
test_that("add endpoint", {
endpoint <- endpoint_add()
res <- endpoint$run(1, 2)
expect_equal(res$data, add(1, 2))
expect_equal(res$status_code, 200)
expect_equal(res$content_type, "application/json")
obj <- api(validate = TRUE)
res_api <- obj$request("GET", "/", list(a = 1, b = 2))
expect_equal(res_api$body, res$body)
expect_equal(res_api$headers[["X-Porcelain-Validated"]], "true")
})
test_that("api fails gracefully when given non-numeric input", {
obj <- api(validate = TRUE)
res <- obj$request("GET", "/", list(a = 1, b = "x"))
expect_equal(res$status, 400)
body <- jsonlite::fromJSON(res$body, simplifyDataFrame = FALSE)
expect_equal(body$status, "failure")
expect_equal(body$errors[[1]]$error, "INVALID_INPUT")
})
Effectively testing such a simple API is quite hard, as we’re not really doing much in the implementation, but these should give an idea of the sorts of strategies that we have used in testing our APIs:
porcelain_response
which you
can inspect to check status code, headers, return value of the target
function and the serialised body. This is typically the workhorse of our
testing.Typically we also include a few light end-to-end tests where we bring
up the API in a separate process and use httr
to interact
with it.