Adam Retter
adam@evolvedbinary.com
XQuery / eXist-db Bootcamp for Humanists
CIHAM and HiSOMA
University of Lyon @ Online
2021-03-22
@adamretter
Using REST
with eXist-db
About Me
-
Director and CTO of Evolved Binary
-
XQuery / XSLT / Schema / RelaxNG
-
Scala / Java / C++ / Rust*
-
Concurrency and Scalability
-
-
Creator of FusionDB multi-model database (5 yrs.)
-
Contributor to Facebook's RocksDB (5 yrs.)
-
Core contributor to eXist-db XML Database (15 yrs.)
-
Founder of EXQuery, and creator of RESTXQ
-
Was a W3C XQuery WG Invited expert
FusionDB, Elemental, and eXist-db
-
What are these, and how are they related?
-
FusionDB
-
Uses eXist-db components - 100% eXist-db API Compatible
-
Improved Storage, ACID Transactions, Instant Backup, etc.
-
Target - Larger projects and organisations
-
Source Available - Free For Dev. / Requires Licenses for Production Use
-
-
Elemental
-
Hard-fork of eXist-db - 100% eXist-db API Compatible (Release Q2 2021)
-
New Features and Fixes. Community Roadmap driven.
-
Target - Smaller projects and users
-
Source Available - Completely Free to Use
-
-
eXist-db
-
No one owns it
-
Open Source - Completely Free to Use
-
-
What we will cover today...
-
What is REST?
-
eXist-db's REST Server
-
Using eXist-db's REST API
-
How to design your own REST API
-
Plain REST vs. controller.xq vs. RESTXQ
What is REST?
-
REST - Representational State Transfer
-
~ Architectural Styles and the Design of Network-based Software Architectures. Fielding, Roy. 2000.
-
A set of Principles, Properties, and Constraints
-
-
Helps guide us to build well-designed Web Applications
-
Focused on the use of HTTP
-
RESTful Applications are applications that adhere to REST principles
-
Not unusual for applications to be partially RESTful
-
You likely use RESTful applications every day via your Web Browser
-
REST - Architectural Constraints
-
Client / Server Architecture
-
Separate the UI concerns from the Storage concerns
-
The Web does this! i.e. Web Browser and Web Server
-
-
Statelessness
-
Every request is isolated and can stand alone
-
-
Cacheability
-
Layered System
-
Code on Demand
-
Uniform Interface
-
URIs, HTTP Headers, and HyperMedia
-
REST - Statelessness
-
Refers to Extrinsic State (Application State)
-
Specifically, NOT keeping Application State at the Server
-
Client maintains its own application state instead
-
Sends what is needed in each request
-
-
Reduces load on the Server
-
No Session needed per-user
-
Reduces conditional computation
-
-
Horizontal Scaling
-
Any server can process any request
-
Keeping Application State at the Server
This breaks Statelessness for REST!
Establish Session
Keeping Application State at the Client
Optional
REST - Uniform Interface
-
URI
-
One per resource (or collection of resources)
-
All apples (or the only apple):
/apple
-
A specific apple:
/apple/pink-lady
-
-
-
Manipulation
-
Verbs = HTTP Methods:
GET
,HEAD
,PUT
,POST
,PATCH
*,DELETE
-
-
Self-Descriptive Messages
-
Media Type
-
HTTP Headers:
Content-Type
,Accept
-
-
Hypermedia (HATEOS)
-
Response should include links to further resources
-
e.g. HTML
-
REST - HTTP Methods
-
GET
-
Retrieve a (representation of a) Resource(s)
-
-
HEAD
-
Get but with no representation (body), just metadata (headers)!
-
-
PUT
-
Store/Replace a Resource at URI
-
-
POST
-
Store a Resource as subordinate of URI
-
Update a Resource (consider
PATCH
instead)
-
-
DELETE
-
Delete a Resource(s)
-
REST - HTTP Headers for Media Types
-
Content Type
-
Indicates the Internet Media Type (Mime Type) of the Request/Response Body
-
e.g. Response from
GET https://www.google.com
:
-
Client POST/PUT/PATCH - tells the server the Media Type of the Resource it is sending
-
Server - tells the client the Media Type of the Resource it is returning
-
-
Accept
-
Indicates one or more Internet Media Types that the client will accept for the Response Body
-
Content-type: text/html; charset=ISO-8859-1
REST - Hypermedia
-
Hyperlinks - Not just HTML!
-
HTML
-
Unbiquitous - understood by your Web-browser
-
<h2>Apples</h2>
<ul>
<li>
<a href="/apple/pink-lady">Pink Lady</a>
</li>
<li>
<a href="/apple/golden-delicious">Golden Delicious</a>
</li>
<li>
<a href="/apple/hollowcore">Hollowcore</a>
</li>
</ul>
-
JSON
[
{"name": "Pink Lady", "href": "/apple/pink-lady"},
{"name": "Golden Delicious", "href": "/apple/golden-delicious"},
{"name": "Hollowcore", "href": "/apple/hollowcore"}
]
REST - Hypermedia
-
XML
-
Syntax for linking is up to you! Various standards exist
-
<apples xmlns:xlink="http://www.w3.org/1999/xlink">
<apple xlink:href="/apple/pink-lady">
<name>Pink Lady</name>
</apple>
<apple xlink:href="/apple/golden-delicious">
<name>Golden Delicious</name>
</apple>
<apple xlink:href="/apple/hollowcore">
<name>Hollowcore</name>
</apple>
</apples>
eXist-db HTTP APIs
eXist-db REST Server
-
API Base URI:
/exist/rest/db
-
REST API for Document Database CRUD Operations
-
Get Collection
-
Create Document
-
Get Document
-
Replace Document
-
Update Document
-
Delete Document
-
Delete Collection
-
-
Additionally
-
Execute Dynamic Query
-
Execute Stored Query
-
eXist-db REST Server - Get Collection
-
HTTP GET
http://localhost:8080/exist/rest/db/
-
Web Browser Example
❯ curl -v -X GET http://localhost:8080/exist/rest/db/
> GET /exist/rest/db/ HTTP/1.1
> Host: localhost:8080
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Sun, 21 Mar 2021 10:50:26 GMT
< Content-Type: application/xml; charset=UTF-8
< Last-Modified: Sun, 21 Mar 2021 10:49:57 GMT
< Created: Sun, 21 Mar 2021 10:49:57 GMT
< Vary: Accept-Encoding, User-Agent
< Transfer-Encoding: chunked
< Server: Jetty(9.4.38.v20210224)
<
<exist:result xmlns:exist="http://exist.sourceforge.net/NS/exist">
<exist:collection name="/db" created="2021-03-21T11:49:56.377+01:00" owner="SYSTEM" group="dba" permissions="rwxr-xr-x">
<exist:collection name="system" created="2021-03-21T11:49:56.381+01:00" owner="SYSTEM" group="dba" permissions="rwxr-xr-x"/>
<exist:collection name="apps" created="2021-03-21T11:49:57.135+01:00" owner="SYSTEM" group="dba" permissions="rwxr-xr-x"/>
</exist:collection>
</exist:result>
-
cURL Example:
eXist-db REST Server - Get Collection
-
HTTP GET
http://localhost:8080/exist/rest/db/
-
Python Example:
#!/usr/bin/python3
import http.client
conn = http.client.HTTPConnection("localhost", 8080)
conn.request("GET", "/exist/rest/db/")
response = conn.getresponse()
print(response.status, response.reason)
responseBody = response.read()
print(responseBody.decode())
conn.close()
200 OK
<exist:result xmlns:exist="http://exist.sourceforge.net/NS/exist">
<exist:collection name="/db" created="2021-03-21T11:49:56.377+01:00" owner="SYSTEM" group="dba" permissions="rwxr-xr-x">
<exist:collection name="system" created="2021-03-21T11:49:56.381+01:00" owner="SYSTEM" group="dba" permissions="rwxr-xr-x"/>
<exist:collection name="apps" created="2021-03-21T11:49:57.135+01:00" owner="SYSTEM" group="dba" permissions="rwxr-xr-x"/>
</exist:collection>
</exist:result>
-
Result:
eXist-db REST Server - Create Document
-
HTTP PUT vs. POST
-
Subtly different
-
Think - Document vs. Collection of Documents
-
Can be confusing at first!
-
-
HTTP PUT - to the URI of the Document
-
e.g.
http://localhost:8080/exist/rest/db/apple/pink-lady.xml
-
NOTE: If document already exists, it will be replaced!
-
-
HTTP POST - to the URI of the Collection in which to store the Document
-
e.g.
http://localhost:8080/exist/rest/db/
-
Response will tell you the new URI of the Document
-
WARNING: eXist-db does NOT support this correctly!
-
eXist-db uses POST only for XQuery and XUpdate.
-
-
eXist-db REST Server - Create (PUT) Doc.
-
HTTP PUT
http://localhost:8080/exist/rest/db/apple/pink-lady.xml
-
cURL Example
❯ curl -v -X PUT -H "Content-Type: application/xml" -d "<apple><name>Pink Lady</name></apple>" \
http://admin:@localhost:8080/exist/rest/db/apple/pink-lady.xml
> PUT /exist/rest/db/apple/pink-lady.xml HTTP/1.1
> Host: localhost:8080
> Authorization: Basic YWRtaW46
> User-Agent: curl/7.64.1
> Accept: */*
> Content-Type: application/xml
> Content-Length: 23
>
< HTTP/1.1 201 Created
< Date: Sun, 21 Mar 2021 11:48:12 GMT
< Content-Length: 0
< Server: Jetty(9.4.38.v20210224)
<
-
TIP: Instead of specifying body inline with
-d
, use a file:
❯ curl -v -X PUT -H "Content-Type: application/xml" -d "@pink-lady.xml" \
http://admin:@localhost:8080/exist/rest/db/apple/pink-lady.xml
eXist-db REST Server - Get Document
-
Similar to Get Collection
-
HTTP GET
http://localhost:8080/exist/rest/db/apple/pink-lady.xml
-
cURL Example
❯ curl -v -X GET http://localhost:8080/exist/rest/db/apple/pink-lady.xml
> GET /exist/rest/db/apple/pink-lady.xml HTTP/1.1
> Host: localhost:8080
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Sun, 21 Mar 2021 12:31:09 GMT
< Last-Modified: Sun, 21 Mar 2021 11:48:13 GMT
< Created: Sun, 21 Mar 2021 11:48:13 GMT
< Content-Type: application/xml; charset=UTF-8
< Vary: Accept-Encoding, User-Agent
< Transfer-Encoding: chunked
< Server: Jetty(9.4.38.v20210224)
<
<apple><name>Pink Lady</name></apple>
eXist-db REST Server - Update Document
-
Uses XUpdate to describe the changes to make
-
NOTE: eXist-db should probably use HTTP PATCH instead of POST!
-
-
HTTP POST
http://localhost:8080/exist/rest/db/apple/pink-lady.xml
-
cURL Example
❯ curl -v -X POST -H "Content-Type: application/xml" \
-d "<xu:modifications version='1.0' xmlns:xu='http://www.xmldb.org/xupdate'>
<xu:update select='/apple/text()'>Just updated</xu:update>
</xu:modifications>" \
http://admin:@localhost:8080/exist/rest/db/apple/pink-lady.xml
> POST /exist/rest/db/apple/pink-lady.xml HTTP/1.1
> Host: localhost:8080
> Authorization: Basic YWRtaW46
> User-Agent: curl/7.64.1
> Accept: */*
> Content-Type: application/xml
> Content-Length: 151
>
< HTTP/1.1 200 OK
< Date: Sun, 21 Mar 2021 12:58:46 GMT
< Content-Type: application/xml; charset=UTF-8
< Transfer-Encoding: chunked
< Server: Jetty(9.4.38.v20210224)
<
<?xml version="1.0" ?><exist:modifications xmlns:exist="http://exist.sourceforge.net/NS/exist" count="1">1 modifications processed.</exist:modifications>
eXist-db REST Server - Update Document
-
Can also be applied to an entire Collection!
-
HTTP POST
http://localhost:8080/exist/rest/db/
-
-
Alternatives:
-
HTTP POST an XQuery containing XQuery Update expression(s)
-
HTTP GET or POST to a Stored XQuery
-
eXist-db REST Server - Delete Document
-
Similar to Get Document
-
HTTP DELETE
http://localhost:8080/exist/rest/db/apple/pink-lady.xml
-
cURL Example
❯ curl -v -X DELETE http://admin:@localhost:8080/exist/rest/db/apple/pink-lady.xml
> DELETE /exist/rest/db/apple/pink-lady.xml HTTP/1.1
> Host: localhost:8080
> Authorization: Basic YWRtaW46
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Sun, 21 Mar 2021 13:12:15 GMT
< Content-Length: 0
< Server: Jetty(9.4.38.v20210224)
<
eXist-db REST Server - Delete Collection
-
Similar to Get Collection
-
HTTP DELETE
http://localhost:8080/exist/rest/db/apple
-
cURL Example
❯ curl -v -X DELETE http://admin:@localhost:8080/exist/rest/db/apple
> DELETE /exist/rest/db/apple HTTP/1.1
> Host: localhost:8080
> Authorization: Basic YWRtaW46
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Sun, 21 Mar 2021 13:15:53 GMT
< Content-Length: 0
< Server: Jetty(9.4.38.v20210224)
<
eXist-db REST Server - Dynamic Query
-
You can either HTTP GET or HTTP POST
-
XQuery Context is the Collection or Document indicated by the URI
-
HTTP GET
http://localhost:8080/exist/rest/db/?_query=/apple
-
Useful for simple/short queries
-
Query and Parameters are sent as URL Query String Parameters
-
Parameters must be URL Encoded
-
Variety of parameters available, e.g.
_start
,_howmany
See eXist-db docs - Get Requests
-
-
HTTP POST
http://localhost:8080/exist/rest/db/
-
Useful for complex/long queries
-
Query and Parameters are sent within HTTP Request Body
-
Variety of parameters available, See eXist-db docs - Post Requests
-
eXist-db REST Server - GET Query
-
cURL Example
❯ curl -v http://localhost:8080/exist/rest/db/?_query=/apple&_start=1&_howmany=10
> GET /exist/rest/db/?_query=/example HTTP/1.1
> Host: localhost:8080
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Sun, 21 Mar 2021 18:08:00 GMT
< Content-Type: application/xml; charset=UTF-8
< Vary: Accept-Encoding, User-Agent
< Transfer-Encoding: chunked
< Server: Jetty(9.4.38.v20210224)
<
<exist:result xmlns:exist="http://exist.sourceforge.net/NS/exist" exist:hits="1" exist:start="1" exist:count="1"
exist:compilation-time="0" exist:execution-time="0">
<apple><name>Pink Lady</name></apple>
</exist:result>
-
TIP: specify
_wrap=no
if you want the raw results
eXist-db REST Server - POST Query
-
cURL Example
❯ curl -v -X POST -H "Content-Type: application/xml" -d "
<query xmlns='http://exist.sourceforge.net/NS/exist' start='1' max='10'>
<text><![CDATA[
/apple
]]></text>
</query>" http://localhost:8080/exist/rest/db/
> POST /exist/rest/db/ HTTP/1.1
> Host: localhost:8080
> User-Agent: curl/7.64.1
> Accept: */*
> Content-Type: application/xml
> Content-Length: 126
>
< HTTP/1.1 200 OK
< Date: Sun, 21 Mar 2021 18:30:06 GMT
< Content-Type: application/xml; charset=UTF-8
< Transfer-Encoding: chunked
< Server: Jetty(9.4.38.v20210224)
<
<exist:result xmlns:exist="http://exist.sourceforge.net/NS/exist" exist:hits="1" exist:start="1" exist:count="1"
exist:compilation-time="1" exist:execution-time="1">
<apple><name>Pink Lady</name></apple>
</exist:result>
-
TIP: specify
wrap="no"
if you want the raw results
eXist-db REST Server - Stored Query
-
You can store an XQuery into the database like any other document!
-
Use the Content-Type:
application/xquery
-
HTTP PUT
http://localhost:8080/exist/rest/db/time.xq
-
cURL Example:
-
❯ curl -v -X PUT -H "Content-Type: application/xquery" -d "<now>{current-dateTime()}</now>" \
http://admin:@localhost:8080/exist/rest/db/time.xq
> PUT /exist/rest/db/time.xq HTTP/1.1
> Host: localhost:8080
> Authorization: Basic YWRtaW46
> User-Agent: curl/7.64.1
> Accept: */*
> Content-Type: application/xquery
> Content-Length: 31
>
< HTTP/1.1 201 Created
< Date: Sun, 21 Mar 2021 18:42:54 GMT
< Content-Length: 0
< Server: Jetty(9.4.38.v20210224)
<
eXist-db REST Server - Stored Query
-
Executable via HTTP GET or POST
-
HTTP GET
http://localhost:8080/exist/rest/db/time.xq
-
cURL Example:
-
❯ curl -v -X GET http://admin:@localhost:8080/exist/rest/db/time.xq
> GET /exist/rest/db/time.xq HTTP/1.1
> Host: localhost:8080
> Authorization: Basic YWRtaW46
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Sun, 21 Mar 2021 18:46:44 GMT
< X-XQuery-Cached: false
< Content-Type: application/xml; charset=UTF-8
< Vary: Accept-Encoding, User-Agent
< Transfer-Encoding: chunked
< Server: Jetty(9.4.38.v20210224)
<
<now>2021-03-21T19:46:44.393+01:00</now>
-
Stored Query can read HTTP Request
e.g.request:get-parameter#2
Designing your own REST API
-
Consider Resources first (the things you talk about)
-
Design URI that point to Resources and Collections of Resources
-
-
Consider Representations (of Resources) second
-
One or more representations, e.g. HTML, XML, JSON, etc.
-
Use HTTP
Content-Type
andAccept
Headers to negotiate
-
-
Consider actions upon those resources
-
Apply HTTP Methods as Verbs to achieve the actions
-
-
Don't forget Hypermedia (i.e. navigability via Hyperlinks)
-
There are Models for evaluating RESTful'ness
Designing your own REST API with eXist-db
-
Consider using RESTXQ first
-
Guides you to create a "RESTful" approach
-
Enforces Statelessness
-
RESTXQ Documentation and Video: http://www.adamretter.org.uk/presentations.xml#xmlprague12
-
-
REST Server with Stored Queries
-
Is just enough to create a REST API of your own
-
Not as pure as RESTXQ
-
REST Server Documentation: http://www.exist-db.org/exist/apps/doc/devguide_rest
-
-
controller.xq
-
Just adds URI Routing/Rewriting to REST Server
-
URL Rewrite Documentation: http://www.exist-db.org/exist/apps/doc/urlrewrite
-
Using REST with eXist-db
By Adam Retter
Using REST with eXist-db
Presentation given for XQuery / eXist-db Bootcamp for Humanists. CIHAM and HiSOMA, University of Lyon - 22 March 2021 - Online
- 1,721