-
Notifications
You must be signed in to change notification settings - Fork 19
Curl Interactions
This page describes a full interaction scenario with rww-play using curl. WARNING: FOR DEVELOPERS only.
curl is a very important tool for developers working on web projects. It is a very lightweight tool for HTTP interactions that allows one to test requests on the web. It is very useful for bug reports, as it makes it easy to set up situations to reproduce bugs without requiring a full browser installation. Since the point of rww-play is that the web is a peer to peer system, where servers can turn into clients and make requests to other servers, we don't want to presuppose a browser installation.
Some issues we have found are:
- the HTTPS connections don't work in WANT mode: curl does not send a certificate if requested politely by the server. At this point it is different from many browsers.
-
OSX Mavericks bug: the
--cert
flag for curl no longer works and you will need to compile your own for reasons explained in this bug report. You can do this by using the Fink Project or Brew.
The data in rww-play is stored as files on the file system,
just as with the main Apache server. The example test_www
directory contains the following files:
$ cd test_www/
$ ls -al
total 40
drwxr-xr-x 2 hjs staff 238 1 Dec 00:47 .
drwxr-xr-x 19 hjs staff 1054 1 Dec 00:03 ..
-rw-r--r-- 1 hjs staff 722 23 Nov 21:35 .acl.ttl
-rw-r--r-- 1 hjs staff 82 23 Nov 21:35 .ttl
lrwxr-xr-x 1 hjs staff 8 23 Nov 21:35 card -> card.ttl
-rw-r--r-- 1 hjs staff 186 28 Nov 13:53 card.acl.ttl
-rw-r--r-- 1 hjs staff 922 1 Dec 00:42 card.ttl
card here is a symbolic link to a default Turtle
representation. Once the server is started ( see the front page,
that file can be fetched using curl with the following curl command:
( the -i
so that we can see the headers returned, and the -k
is there
so that we don't have trouble with the localhost non certified certificate
our server is using on the https
port 8443)
$ curl -i -k https://localhost:8443/2013/card
HTTP/1.1 200 OK
Accept-Patch: application/sparql-update
Access-Control-Allow-Origin: *
Allow: OPTIONS, GET, HEAD, SEARCH
Content-Type: text/turtle
ETag: "1417390950000|Success(922)"
Last-Modified: Sun, 30 Nov 2014 23:42:30 GMT
Link: <http://www.w3.org/ns/ldp#Resource>; rel=type, <card.acl>; rel=acl
Content-Length: 978
<#me> <http://xmlns.com/foaf/0.1/knows> <http://bblfish.net/people/henry/card#me> .
_:node1981hgchsx1 a <http://www.w3.org/ns/auth/cert#RSAPublicKey> ;
<http://www.w3.org/ns/auth/cert#modulus> "b7cb16af0aeec58a4c0c05e0504a334382a1db7a8a092057f97c271439f7ff8cfd469b615934fa401b4b320b756cf017e16c8ee0d5afceed1a54390738720c67813b765e1bf9e310809e133b7f7c2aca34e185c3bdcd42fc40d84772ad691f36b9078c8e0079f64089ae0adcaa80d4186cf683403d6485e578dbde161a82b4e34650cb77fd274fe84bb7ae488a3236f146178cf836cc701b1d3c40c0d7a8e838afc209e3b5c825fa9702017b52492f4cf4bdeb089726e2778eb63b8854c8b366b2c5425f5dec236c02c8e760b7303adfb2a94bf835c2e28901abeca292d7ca04c1ae3c377e2d2f3e011be76868d941678a18c2abf78f98f796f493f2a946cf2d"^^<http://www.w3.org/2001/XMLSchema#hexBinary> ;
<http://www.w3.org/ns/auth/cert#exponent> 65537 .
<#me> <http://www.w3.org/ns/auth/cert#key> _:node1981hgchsx1 ;
<http://xmlns.com/foaf/0.1/name> "Me (StampleIO test webid)" ;
a <http://xmlns.com/foaf/0.1/Person> .
The server knows how to do content-negotiation, and can also serve the
file in other formats such as Json-Ld.
We do this by telling the server that we accept that mime type in a header
that we add with the -H "Accept: application/ld+json"
parameter to curl
$ curl -i -H "Accept: application/ld+json" -k https://localhost:8443/2013/card
HTTP/1.1 200 OK
Accept-Patch: application/sparql-update
Access-Control-Allow-Origin: *
Allow: OPTIONS, GET, HEAD, SEARCH
Content-Type: application/ld+json
ETag: "1417390950000|Success(922)"
Last-Modified: Sun, 30 Nov 2014 23:42:30 GMT
Link: <http://www.w3.org/ns/ldp#Resource>; rel=type, <card.acl>; rel=acl
Content-Length: 1245
{
"@graph" : [ {
"@id" : "#me",
"@type" : "http://xmlns.com/foaf/0.1/Person",
"http://www.w3.org/ns/auth/cert#key" : {
"@id" : "_:node1981hgchsx1"
},
"http://xmlns.com/foaf/0.1/knows" : {
"@id" : "http://bblfish.net/people/henry/card#me"
},
"http://xmlns.com/foaf/0.1/name" : "Me (StampleIO test webid)"
}, {
"@id" : "_:node1981hgchsx1",
"@type" : "http://www.w3.org/ns/auth/cert#RSAPublicKey",
"http://www.w3.org/ns/auth/cert#exponent" : {
"@type" : "http://www.w3.org/2001/XMLSchema#integer",
"@value" : "65537"
},
"http://www.w3.org/ns/auth/cert#modulus" : {
"@type" : "http://www.w3.org/2001/XMLSchema#hexBinary",
"@value" : "b7cb16af0aeec58a4c0c05e0504a334382a1db7a8a092057f97c271439f7ff8cfd469b615934fa401b4b320b756cf017e16c8ee0d5afceed1a54390738720c67813b765e1bf9e310809e133b7f7c2aca34e185c3bdcd42fc40d84772ad691f36b9078c8e0079f64089ae0adcaa80d4186cf683403d6485e578dbde161a82b4e34650cb77fd274fe84bb7ae488a3236f146178cf836cc701b1d3c40c0d7a8e838afc209e3b5c825fa9702017b52492f4cf4bdeb089726e2778eb63b8854c8b366b2c5425f5dec236c02c8e760b7303adfb2a94bf835c2e28901abeca292d7ca04c1ae3c377e2d2f3e011be76868d941678a18c2abf78f98f796f493f2a946cf2d"
}
} ]
}
Those two representations are required by LDP, but we also provide others such as
the older text/rdf+xml
. Try it out!
Both of these representations determine exactly the same graph: They mean exactly the same thing, i.e. all worlds in which the one is true, the other is true too.
At the top of the previous file we could find the following
Link: <http://www.w3.org/ns/ldp#Resource>; rel=type, <card.acl>; rel=acl
This comes from the Web Linking RFC 5988. It
described the resource with two relations, separated by the somewhat counterintuitive ,
character, which is usually thought to be weaker than ;
- but not in HTTP headers!
The two relations are:
-
<> type <http://www.w3.org/ns/ldp#Resource>
: the resource we got is an ldp:Resource -
<> acl <card.acl>
: the Acl for this resource is to be found at the relative url card.acl
card.acl
is the Web Access Control graph for that resource. We can find it on the file system
$ cat card.acl.ttl
@prefix acl: <http://www.w3.org/ns/auth/acl#> .
[] acl:accessTo <card>, <card.ttl> ;
acl:agentClass <http://xmlns.com/foaf/0.1/Agent> ;
acl:mode acl:Read .
<> acl:include <.acl> .
The acl for card
just includes the acl for the directory/collection .
(TODO: wac:include
has not yet been defined in the Web Access Control Ontology)
$ cat card.acl.ttl
@prefix wac: <http://www.w3.org/ns/auth/acl#> .
@prefix foaf: <http://xmlns.com/foaf/0.1/> .
<> wac:include <.acl> .
You can also get the same file with curl with the command
$ curl -i -k https://localhost:8443/2013/card.acl
That file includes all the acts from the directory acl. And that says
$ cat .acl.ttl
@prefix acl: <http://www.w3.org/ns/auth/acl#> .
@prefix foaf: <http://xmlns.com/foaf/0.1/> .
[] acl:accessToClass [ acl:regex "https://(\\w+\\.)?localhost:8443/.*[.]acl" ];
acl:mode acl:Read;
acl:agentClass foaf:Agent .
[] acl:accessToClass [ acl:regex "https://(\\w+\\.)?localhost:8443/.*" ];
acl:mode acl:Write, acl:Read;
acl:agent <card#me> .
This says that all files on this server that refer to this ACL in their headers via the
ACL relation are readable by anyone ( foaf:Agent ), and writeable by the Agent
identified by the WebID whose relative uri is <card#me>
i.e., since it is taken
from the resource https://localhost:8443/2013/
by following the URI specification on
relative URI resolution
it is the agent referred to by <https://localhost:8443/2013/card#me>
( for more details see
WebID Specification
So this is a quite usual setup: all the files on a web server a readable by everyone but only writeable by the administrator.
Let's try that out. What happens when we try to PATCH the file. Say we want to change the name from the current "Me (StampleIO test webid)" to say "Henry Story". In the example directory we have a SPARQL Updated example
$ cat ../eg/card.update
PREFIX foaf: <http://xmlns.com/foaf/0.1/>
DELETE DATA { <https://localhost:8443/2013/card#me> foaf:name "Me (StampleIO test webid)" . };
PREFIX foaf: <http://xmlns.com/foaf/0.1/>
INSERT DATA { <https://localhost:8443/2013/card#me> foaf:name "Henry Story" . }
( I find it a bit odd that in SPARQL Update one has to write the PREFIX out twice, but you can submit comments for improvements )
If we try to use the HTTP PATCH verb that the Linked Data Platform suggests we use then we get the following:
$ curl -X PATCH -k -i --data-binary @../eg/card.update \
-H "Content-Type: application/sparql-update; utf-8" \
https://localhost:8443/2013/card
curl: (56) SSL read: error:14094412:SSL routines:SSL3_READ_BYTES:sslv3 alert bad certificate, errno 0
We get ( only with curl ) a bizaare error. This is because curl does not allow WANT
TLS connections, but only NEED, and so a failure to authenticate closes the connection. Most browsers have the more friendly behaviour allowing the server to return an HTTP error code and an explanatory body.
Requesting the same resource with the right client certificate we get a more interesting answer
$ curl -X PATCH -k -i --data-binary @../eg/card.update \
-H "Content-Type: application/sparql-update; utf-8" \
--cert ../eg/test-localhost.pem:test https://localhost:8443/2013/card
HTTP/1.1 428 Precondition Required
Content-Type: text/plain; charset=utf-8
Content-Length: 38
altering the resource requires an ETag
The client certificate in ../eg/test-localhost.pem
contains exactly the public key given in the above
cert as you can see by comparing the modulus and exponents in both representations. This
is what allows the authentication to go through, using the WebID over TLS protocol.
$ $ openssl x509 -in ../eg/test-localhost.pem -inform pem -text
Certificate:
Data:
Version: 3 (0x2)
Serial Number:
01:49:f1:d4:3d:e6
Signature Algorithm: sha1WithRSAEncryption
Issuer: CN=WebID, O={}
Validity
Not Before: Nov 27 15:07:38 2014 GMT
Not After : Nov 24 15:17:38 2024 GMT
Subject: dnQualifier=tester@localhost.edu
Subject Public Key Info:
Public Key Algorithm: rsaEncryption
RSA Public Key: (2048 bit)
Modulus (2048 bit):
00:b7:cb:16:af:0a:ee:c5:8a:4c:0c:05:e0:50:4a:
33:43:82:a1:db:7a:8a:09:20:57:f9:7c:27:14:39:
f7:ff:8c:fd:46:9b:61:59:34:fa:40:1b:4b:32:0b:
75:6c:f0:17:e1:6c:8e:e0:d5:af:ce:ed:1a:54:39:
07:38:72:0c:67:81:3b:76:5e:1b:f9:e3:10:80:9e:
13:3b:7f:7c:2a:ca:34:e1:85:c3:bd:cd:42:fc:40:
d8:47:72:ad:69:1f:36:b9:07:8c:8e:00:79:f6:40:
89:ae:0a:dc:aa:80:d4:18:6c:f6:83:40:3d:64:85:
e5:78:db:de:16:1a:82:b4:e3:46:50:cb:77:fd:27:
4f:e8:4b:b7:ae:48:8a:32:36:f1:46:17:8c:f8:36:
cc:70:1b:1d:3c:40:c0:d7:a8:e8:38:af:c2:09:e3:
b5:c8:25:fa:97:02:01:7b:52:49:2f:4c:f4:bd:eb:
08:97:26:e2:77:8e:b6:3b:88:54:c8:b3:66:b2:c5:
42:5f:5d:ec:23:6c:02:c8:e7:60:b7:30:3a:df:b2:
a9:4b:f8:35:c2:e2:89:01:ab:ec:a2:92:d7:ca:04:
c1:ae:3c:37:7e:2d:2f:3e:01:1b:e7:68:68:d9:41:
67:8a:18:c2:ab:f7:8f:98:f7:96:f4:93:f2:a9:46:
cf:2d
Exponent: 65537 (0x10001)
X509v3 extensions:
X509v3 Subject Alternative Name: critical
URI:https://localhost:8443/2013/card#me
X509v3 Key Usage: critical
Digital Signature, Non Repudiation, Key Encipherment, Key Agreement, Certificate Sign
X509v3 Basic Constraints: critical
CA:FALSE
Netscape Cert Type:
SSL Client, S/MIME
Signature Algorithm: sha1WithRSAEncryption
95:da:39:18:00:a5:7a:16:4f:cd:d2:b8:21:97:0e:e5:c7:20:
c1:50:21:66:e3:63:31:cf:72:f0:5b:9f:8d:57:a3:98:4f:21:
0f:a7:1c:3e:a3:39:64:e7:e4:ec:29:48:f7:a6:d3:fb:9c:99:
44:a3:44:12:3d:06:57:62:9b:9d:30:9a:7c:3c:35:6d:59:e3:
6e:3e:7a:e7:86:44:64:1e:16:04:8d:69:d1:f2:c4:05:e6:9b:
7b:f2:a4:cf:48:da:78:06:78:ff:14:be:90:b2:f7:8a:5d:ac:
55:da:18:25:c8:45:f1:7b:3e:f2:ab:c5:1f:13:5f:3c:9d:16:
a1:a8:5e:8f:4b:0c:ec:f8:71:4a:b5:86:4f:db:cd:87:c1:99:
75:9b:ff:34:4f:dc:da:ed:61:14:95:85:d5:6f:b3:c5:68:90:
9a:9f:32:23:1f:19:00:25:8c:6e:88:42:de:ad:2d:94:41:7a:
c4:96:6d:9f:68:a3:2f:4c:6c:99:de:6d:de:66:0f:84:fc:87:
9c:59:a7:d1:78:3a:5d:8d:75:32:93:a1:34:0c:b8:30:0f:ec:
9b:32:cc:90:b7:13:3d:a4:1f:3b:67:9a:74:ac:27:00:ed:ce:
0d:32:9b:f4:37:b2:18:ba:c9:49:a6:97:0c:e6:9c:e4:e9:48:
00:8f:df:3f
-----BEGIN CERTIFICATE-----
This server does not allow a patch without an If-Match
header.
( this should be something configurable ). When we made our original
request to the card it came with an tag
ETag: "1417390950000|Success(922)"
so we need to use this in our PATCH request
$ curl -X PATCH -k -i --data-binary @../eg/card.update \
-H "Content-Type: application/sparql-update; utf-8" \
-H 'If-Match: "1417390950000|Success(922)"' \
--cert ../eg/test-localhost.pem:test \
https://localhost:8443/2013/card
HTTP/1.1 200 OK
Content-Type: text/plain; charset=utf-8
User: https://localhost:8443/2013/card#me
Content-Length: 9
Succeeded
And indeed it worked as shown by the 200 Ok
response. The card should now have the new name.
Note that the original GET on the resource gave us the Allow header with the following values
Allow: OPTIONS, GET, HEAD, SEARCH
and that this did not include PATCH
, nor DELETE
for the matter. That is because neither
those methods are allowed by an anonymous user. An authenticated user would find those methods
also listed. ( this is easier to verify in a browser )
Next we can publish a couch surfing opportunity using HTTP's POST method as explained by the LDP spec
$ curl -X POST -k -i -H "Content-Type: text/turtle; utf-8" \
--cert ../eg/test-localhost.pem:test \
-H "Slug: couch" -d @../eg/couch.ttl https://localhost:8443/2013/ \
HTTP/1.1 201 Created
Location: https://localhost:8443/2013/couch
Content-Length: 0
The Location:
header in the above response tells us that the name of the created resource is https://localhost:8443/2013/couch.
We can now look at the contents of the https://localhost:8443/2013/ collection, where we
should see - and do - the new couch
resource listed as having been created by ldp.
$ curl -k -i -X GET \
--cert ../eg/test-localhost.pem:test https://localhost:8443/2013/
HTTP/1.1 200 OK
Link: <https://localhost:8443/2013/.acl>; rel=acl
Content-Type: text/turtle
Content-Length: 701
<> <http://xmlns.com/foaf/0.1/maker> <http://bblfish.net/people/henry/card#me> ;
<http://www.w3.org/ns/ldp#created> <card> , <test/> , <couch> .
<test/> <http://www.w3.org/ns/posix/stat#mtime> "1382465690000"^^<http://www.w3.org/2001/XMLSchema#integer> ;
a <http://www.w3.org/ns/ldp#Container> .
<couch> <http://www.w3.org/ns/posix/stat#mtime> "1382465783000"^^<http://www.w3.org/2001/XMLSchema#integer> ;
<http://www.w3.org/ns/posix/stat#size> "9"^^<http://www.w3.org/2001/XMLSchema#integer> .
<card> <http://www.w3.org/ns/posix/stat#size> "8"^^<http://www.w3.org/2001/XMLSchema#integer> ;
<http://www.w3.org/ns/posix/stat#mtime> "1372357799000"^^<http://www.w3.org/2001/XMLSchema#integer> .
Each resource in the LDPC is listed and the size of its Turtle representation on disk is given.
Notice also that the LDPC </2013>
is described as having been made by Henry Story. Such information is stored in the
test_www/.ttl file, which can also be edited using the usual LDP protocol. (TODO: is this ok with LDP?)
We can find out about the ACL for this resource using HEAD (TODO: OPTIONS would be better, but is not implemented yet )
$ curl -X HEAD -k -i --cert ../eg/test-localhost.pem:test \
https://localhost:8443/2013/couch
HTTP/1.1 200 OK
Link: <https://localhost:8443/2013/couch.acl>; rel=acl
Content-Type: text/turtle
Content-Length: 0
( TODO: The Content-Length should not be 0 for HEAD. Bug in Play2.0 probably )
So we add the couch acl which gives access to that information in addition to the default owner of the collection, to two groups of people by patching the acl with couch.acl.patch
$ curl -X PATCH -k -i -H "Content-Type: application/sparql-update; utf-8" \
--cert ../eg/test-localhost.pem:test \
--data-binary @../eg/couch.acl.patch \
https://localhost:8443/2013/couch.acl
HTTP/1.1 200 OK
Content-Type: text/plain; charset=utf-8
Content-Length: 9
Succeeded
This makes it available to the test user and the members of the WebID and OuiShare groups. If you have a WebID then try
adding yours and test it. You can also request different formats by changing the Accept:
header such as with the following request for RDF/XML
$ curl -k -i -X GET -H "Accept: application/rdf+xml" --cert ../eg/test-localhost.pem:test https://localhost:8443/2013/couch
or if you prefer rdf/xml over turtle as described by the Content Negotiation section of the HTTP1.1 spec:
$ curl -k -i -X GET -H "Accept:application/rdf+xml;q=0.9,text/turtle;q=0.7" --cert ../eg/test-localhost.pem:test https://localhost:8443/2013/couch
You can also upload binary resources though at present the mime types
uplowdable are artificially limited, and it just requires one to
improve the SupportedBinaryMimeExtensions
object.
Here is how you can upload that Apache feather logo to your web site:
curl -X POST -k -i -H "Content-Type: image/png" \
--cert ../eg/test-localhost.pem:test \
-H "Slug: Apache-feather"
--data-binary @../eg/apache-feather.png
https://localhost:8443/2013/
It is then possible to also use SPARQL queries on particular resources. (TODO: find a better example)
$ cat ../eg/couch.sparql
PREFIX gr: <http://purl.org/goodrelations/v1#>
SELECT ?D
WHERE {
[] a <http://bblfish.net/2013/05/10/couch#Surf>;
gr:description ?D .
}
$ curl -X SEARCH -k -i -H "Content-Type: application/sparql-query; charset=UTF-8" \
--cert ../eg/test-localhost.pem:test \
--data-binary @../eg/couch.sparql https://localhost:8443/2013/couch
HTTP/1.1 200 OK
Content-Type: application/sparql-results+xml
Content-Length: 337
<?xml version='1.0' encoding='UTF-8'?>
<sparql xmlns='http://www.w3.org/2005/sparql-results#'>
<head>
<variable name='D'/>
</head>
<results>
<result>
<binding name='D'>
<literal datatype='http://www.w3.org/2001/XMLSchema#string'>Comfortable couch in Artist Stables</literal>
</binding>
</result>
</results>
</sparql>
You can also send a DESCRIBE query ( which sadly is not very well defined, so that it may not return the same with different SPARQL engines. This is what is returned with the sesame engine:
$ cat ../eg/card-describe.sparql
DESCRIBE <https://localhost:8443/2013/card#me>
$ curl -X SEARCH -k -i -H "Content-Type: application/sparql-query; charset=UTF-8" --cert ../eg/test-localhost.pem:test --data-binary @../eg/card-describe.sparql https://localhost:8443/2013/card
HTTP/1.1 200 OK
Content-Type: text/turtle
User: http://xmlns.com/foaf/0.1/Agent
Content-Length: 1121
<https://localhost:8443/2013/card#me> a <http://xmlns.com/foaf/0.1/Person> ;
<http://xmlns.com/foaf/0.1/name> "Me (StampleIO test webid)" ;
<http://xmlns.com/foaf/0.1/knows> <http://bblfish.net/people/henry/card#me> ;
<http://www.w3.org/ns/auth/cert#key> _:node199fktmr7x1 .
_:node199fktmr7x1 a <http://www.w3.org/ns/auth/cert#RSAPublicKey> ;
<http://www.w3.org/ns/auth/cert#exponent> 65537 ;
<http://www.w3.org/ns/auth/cert#modulus> "b7cb16af0aeec58a4c0c05e0504a334382a1db7a8a092057f97c271439f7ff8cfd469b615934fa401b4b320b756cf017e16c8ee0d5afceed1a54390738720c67813b765e1bf9e310809e133b7f7c2aca34e185c3bdcd42fc40d84772ad691f36b9078c8e0079f64089ae0adcaa80d4186cf683403d6485e578dbde161a82b4e34650cb77fd274fe84bb7ae488a3236f146178cf836cc701b1d3c40c0d7a8e838afc209e3b5c825fa9702017b52492f4cf4bdeb089726e2778eb63b8854c8b366b2c5425f5dec236c02c8e760b7303adfb2a94bf835c2e28901abeca292d7ca04c1ae3c377e2d2f3e011be76868d941678a18c2abf78f98f796f493f2a946cf2d"^^<http://www.w3.org/2001/XMLSchema#hexBinary> .
<https://localhost:8443/2013/card> <http://xmlns.com/foaf/0.1/primaryTopic> <https://localhost:8443/2013/card#me> .
Note how this returns the symmetric concise bounded description, that is the subgraph of the graph consisting of all arcs going from the and into original URI, and the arc going from those nodes if the nodes are blank nodes, until it reaches either literals or URIs.
Finally if you no longer want the couch surfing opportunity to be published you can DELETE
it.
( It would be better to express that it was sold: DELETE
ing resources on the Web is usually
bad practice: it breaks the links that other people set up to your services )
curl -k -i -X DELETE -H "Accept: text/turtle"
--cert ../eg/test-localhost.pem:test https://localhost:8443/2013/couch
HTTP/1.1 200 OK
Content-Length: 0
And so the resource no longer is listed in the LDPC
$ curl -k -i -X GET \
--cert ../eg/test-localhost.pem:test \
https://localhost:8443/2013/
HTTP/1.1 200 OK
Link: <https://localhost:8443/2013/.acl>; rel=acl
Content-Type: text/turtle
Content-Length: 109
<> a <http://www.w3.org/ns/ldp#Container> ;
<http://www.w3.org/ns/ldp#created> <card> , <raw/> , <test/> .
To create a new container one just creates an LDP Resource that contains the triple <> a ldp:Container
.
$ cat ../eg/newContainer.ttl
@prefix ldp: <http://www.w3.org/ns/ldp#> .
@prefix foaf: <http://xmlns.com/foaf/0.1/> .
<> a ldp:Container;
foaf:topic "A container for some type X of resources";
foaf:maker <../card#me> .
$ curl -X POST -k -i -H "Content-Type: text/turtle; utf-8" --cert ../eg/test-localhost.pem:test -H "Slug: XDir" -d @../eg/newContainer.ttl https://localhost:8443/2013/
HTTP/1.1 201 Created
Location: https://localhost:8443/2013/XDir/
Content-Length: 0
Note that directories can only be deleted if all ldp:created
resources were previously deleted.