Network Working Group | J. Reschke |
Internet-Draft | greenbytes |
Intended status: Informational | August 9, 2006 |
Expires: February 10, 2007 |
By submitting this Internet-Draft, each author represents that any applicable patent or other IPR claims of which he or she is aware have been or will be disclosed, and any of which he or she becomes aware will be disclosed, in accordance with Section 6 of BCP 79.¶
Internet-Drafts are working documents of the Internet Engineering Task Force (IETF), its areas, and its working groups. Note that other groups may also distribute working documents as Internet-Drafts.¶
Internet-Drafts are draft documents valid for a maximum of six months and may be updated, replaced, or obsoleted by other documents at any time. It is inappropriate to use Internet-Drafts as reference material or to cite them other than as “work in progress”.¶
The list of current Internet-Drafts can be accessed at http://www.ietf.org/ietf/1id-abstracts.txt.¶
The list of Internet-Draft Shadow Directories can be accessed at http://www.ietf.org/shadow.html.¶
This Internet-Draft will expire on February 10, 2007.¶
Copyright © The Internet Society (2006). All Rights Reserved.¶
The Hypertext Transfer Protocol (HTTP) specifies a state identifier, called "Entity Tag", to be returned in the "ETag" response header. However, the description of this header for write operations such as PUT is incomplete, and has caused confusion among developers and protocol designers, and potentially interoperability problems.¶
This document explains the problem in detail and suggests both a clarification for a revision to the HTTP/1.1 specification (RFC2616) and a new response header, making HTTP entity tags more useful for user agents that want to avoid round-trips to the server after modifying a resource.¶
Distribution of this document is unlimited. Please send comments to the Hypertext Transfer Protocol (HTTP) mailing list at ietf-http-wg@w3.org, which may be joined by sending a message with subject "subscribe" to ietf-http-wg-request@w3.org.¶
Discussions of the HTTP working group are archived at <http://lists.w3.org/Archives/Public/ietf-http-wg/>.¶
XML versions, latest edits and the issues list for this document are available from <http://greenbytes.de/tech/webdav/#draft-reschke-http-etag-on-write-latest>.¶
The Hypertext Transfer Protocol (HTTP) specifies a state identifier, called "Entity Tag", to be returned in the "ETag" response header (see [RFC2616], Section 14.19). However, the description of this header for write operations such as PUT is incomplete, and has caused confusion among developers and protocol designers, and potentially interoperability problems.¶
This document explains the problem in detail and suggests both a clarification for a revision to [RFC2616] and a new response header, making HTTP entity tags more useful for user agents that want to avoid round-trips to the server after modifying a resource.¶
Note that there is a related problem: modifying content-negotiated resources. Here the consensus seems to be simply not to do it. Instead, the origin server should reveal specific URIs of content that is not content-negotiated in the Content-Location response header ([RFC2616], Section 14.14), and user agents should use this more specific URI for authoring. Thus, the remainder of this document will focus on resources for which no content negotiation takes place.¶
Another related question is the usability of the weak entity tags for authoring (see [RFC2616], Section 13.3.3). Although this document focuses on the usage of strong entity tags, it is believed that the changes suggested in this document could be applied to weak entity tags as well.¶
For a long time, nobody realized that there was a problem at all, or those who realized preferred to ignore it.¶
Server implementers added code that would return the new value of the "ETag" header in a response to a successful PUT request. After all, the client could be interested in it.¶
User agent developers in turn were happy to get a new "ETag" value, saving a subsequent HEAD request to retrieve the new entity tag.¶
However, at some point of time, potentially during a Web Distributed Authoring and Versioning (WebDAV, [RFC2518]) interoperability event, client programmers asked servers to always return "ETag" headers upon PUT, never ever to change the entity tag without "good reason", and - by the way - always to guarantee that the server stores the new content octet-by-octet.¶
From the perspective of client software that wants to treat an HTTP server as a file system replacement, this makes a lot of sense. After all, when one writes to a file one usually expects the file system to store what was written, and not to unexpectedly change the contents.¶
However, in general, an HTTP server is not a file system replacement. There may be some that have been designed that way, and some that expose some parts of their namespace that have this quality. But in general, HTTP server implementers have a lot of freedom in how resources are implemented. Indeed, this flexibility is one of the reasons for HTTP's success, allowing to be used for a wide range of tasks, of which replacing file systems is just one (and not necessarily the most interesting one).¶
In particular: ¶
Furthermore: ¶
As long as servers store the content octet-by-octet, and return exactly what the client wrote, there is no problem at all.¶
Things get more interesting when a server does change the content, such as in the "simple authoring" example given in Appendix A.1. Here, the server does change the content upon writing to the resource, yet no harm is done, because the final state of the resource on the server does not depend on the client being aware of that.¶
All of the content rewriting examples mentioned above have this quality: the client can safely continue to edit the entity it sent, because the result of the transformation done by the server will be the same in the end. Formally, if we call the server-side transformation "t", the initial content "c", and the client-side editing steps "e1" and "e2", then¶
t(e2(e1(c))) = t(e2(t(e1(c))))
e.g., it's irrelevant whether the client obtained the current entity body before doing its second edit.
[example.for.non.safe.rewrite: Question: does anybody know a real-world example for server-side content rewriting where the above is not true? --julian.reschke@greenbytes.de] ¶
Problems will only occur if the client uses the entity body it sent, and the entity tag it obtained in return, in subsequent requests that only transfer part of the entity body, such as GET or PUT requests using the "Range" request header (see [RFC2616], Section 14.35).¶
Furthermore, some clients need to expose the actual contents to the end user. These clients will have to ensure that they really have the current representation.¶
Entity bodies (and thus entity tags) changing due to side effects of seemingly unrelated requests are indeed a problem, as demonstrated in Appendix A.2, and this specification proposes a way to resolve this in Section 3.¶
There are several places in the HTTP/1.1 specification ([RFC2616]) mentioning the "ETag" response header.¶
Let us start with the header definition in Section 14.19:¶
The meaning of a "response-header" in turn is defined in Section 6.2:¶
The response-header fields allow the server to pass additional information about the response which cannot be placed in the Status-Line. These header fields give information about the server and about further access to the resource identified by the Request-URI.¶
The "ETag" response header itself is mentioned mainly in the context of cache validation, such as in Section 13.3.2. What is missing is a coherent description on how the origin server can notify the user-agent when the entity tag changes as result of a write operation, such as PUT.¶
Indeed, the definition of the 201 Created status code mentions entity tags (Section 10.2.2):¶
The "ETag" response header is mentioned again in the definition of 206 Partial Content (Section 10.2.7) and 304 Not Modified (Section 10.3.5), but notably missing are statements about other 2xx series status codes that can occur upon a successful PUT operation, such as 200 OK (Section 10.2.1) and 204 No Content (Section 10.2.5).¶
While working on the revision of [RFC2518], the IETF WebDAV working group realized that this is a generic problem that needs attention independently of WebDAV. An initial attempt was made with [draft-whitehead-http-etag] in February 2006, but no progress was made since.¶
At the time of this writing in August 2006, two specifications based on HTTP were under IESG review, taking two opposite approaches: ¶
In essence, this makes it impossible to implement an HTTP resource that conforms to both specifications. Due to the differing use cases of XCAP and CalDAV, this may not be a problem in practice, but the disagreement in itself is scary. Publication of these specifications on the standards track will make it much harder for future protocols to deal with this topic in a meaningful way (comments were sent during IETF Last Call for CalDAV, see <http://www1.ietf.org/mail-archive/web/ietf/current/msg43021.html>).¶
The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL-NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in [RFC2119].¶
The terminology used here follows and extends that in the HTTP specification [RFC2616], notably the augmented Backus-Naur Form (BNF) defined in Section 2.1 of that document.¶
This section describes a minimal change to [RFC2616], proposed in <http://lists.w3.org/Archives/Public/ietf-http-wg/2006JanMar/0003.html>.¶
At the end of Section 10.2 of [RFC2616], add:¶
The response MAY contain an ETag response header field indicating the current value of the entity tag (Section 14.19) for the requested variant. The value SHOULD be the same as the one returned upon a subsequent HEAD request addressing the same variant.¶
In Section 10.2.1 of [RFC2616], remove:¶
A 201 response MAY contain an ETag response header field indicating the current value of the entity tag for the requested variant just created, see Section 14.19.¶
In essence, this moves the statement about entity tags in write operations from the specific case of status 201 Created into the more generic description of the 2xx series status codes.¶
[rcf2616.enhancements: Suggest other enhancements, such as actually recommending returning updated entity tags in all write operations? Also, should "requested variant" be clarified in the context of write operations? --julian.reschke@greenbytes.de] ¶
The "Entity-Transform" response header can be used by the origin server in responses to any request that affects the entity body of a resource. It specifies in which way the server has modified the entity before storage, thus allowing the user agent to decide about whether it needs to refresh a local copy.¶
Entity-Transform = "Entity-Transform" ":" 1#transform-info transform-info = "identity" | "unspecified" | token | coded-URI coded-URI = "<" absolute-URI ">" ; token: [RFC2616], Section 2.2 ; absolute-URI: [RFC3986], Section 4.3
A transform-info of "identity" specifies that the origin server has stored the entity octet-by-octet, thus the user agent MAY use a local copy for subsequent requests that rely on octet-by-octet identity (such as a PUT with "Range" request header). Servers MUST NOT return "identity" with any other transform-info values.¶
Both the absence of this response header and any value other than "identity" specify that the origin server may have transformed the entity before storage, thus a subsequent retrieval will not necessarily return an exact copy of the stored entity.¶
Future IETF specifications may define other token values for more precise information about how the content was rewritten. Values taking the form of a coded-URI can be defined without registration; in this case, it is RECOMMENDED to choose a URI that allows retrieval of a human-readable description.¶
See [RFC2616], Section 6.2 for a definition of "response header".¶
Content was stored octet-by-octet:
Entity-Transform: identity
Content was transformed, but server doesn't specify how:
Entity-Transform: unspecified
Content was transformed, and server specifies the kind of transformation:
Entity-Transform: <http://example.org/transformations/xml>
The clarification of [RFC2616] (see Section 3) makes it clear that user agents can use "ETag" headers obtained in write operations, as long as they do not require octet-by-octet identity. In particular, a new entity tag can be returned for any method, such as a WebDAV PROPPATCH (see [RFC2518], Section 8.2. This helps dealing with the problem described in Appendix A.2. See Appendix A.3 for details.¶
The addition of the "Entity-Transform" response header (see Section 4) enables origin servers to signal that they stored an exact copy of the content, thus allowing clients not to refetch the content. Note that by default (in absence of the response header), a client can not make any assumptions about the server's behavior in this regard. Thus clients will only benefit from new servers explicitly setting the new header.¶
This document specifies the new HTTP response header listed below.¶
This specification introduces no new security considerations beyond those discussed in Section 15 of [RFC2616].¶
In this example, the server exposes data extracted from the HTML <title> element ([HTML], Section 7.4.2) as a custom WebDAV property ([RFC2518], Section 4), allowing both read and write access.¶
In the first step, the client obtains the current representation for <http://example.com/test.html>:¶
>> Request (1)
GET /test.html HTTP/1.1 Host: example.com
>> Response (1)
HTTP/1.1 200 OK Content-Type: text/html ETag: "A" <html> <head> </head> <body> </body> </html>
Next, it adds one paragraph to the <body> element, and gets back a new entity tag:¶
>> Request (2)
PUT /test.html HTTP/1.1 Host: example.com If-Match: "A" <html> <head> </head> <body> <p>First paragraph.</p> </body> </html>
>> Response (2)
HTTP/1.1 200 OK ETag: "B"
Next, the client sets a custom "title" property (see [RFC2518], Section 8.2):¶
>> Request (3)
PROPPATCH /test.html HTTP/1.1 Host: example.com Content-Type: application/xml <proppatch xmlns="DAV:"> <set> <prop> <title xmlns="http://ns.example.org/"> A sample document </title> </prop> </set> </proppatch>
>> Response (3)
HTTP/1.1 207 Multi-Status Content-Type: application/xml <multistatus xmlns="DAV:"> <response> <href>/test.html</href> <propstat> <prop> <title xmlns="http://ns.example.org/"/> </prop> <status>HTTP/1.1 2OO OK</status> </propstat> </response> </multistatus>
The server knows how to propagate property changes into the HTML content, so it updates the entity by adding an HTML title document accordingly. This causes the entity tag changing to "C".¶
The new entity body is shown below, but the client does not realize that it did change at all.
<html> <head> <title>A sample document</title> </head> <body> <p>First paragraph.</p> </body> </html>
A subsequent attempt by the client to update the entity body will fail, unless it realizes that changing WebDAV properties may affect the entity as well. In this case, it would have had to get the current entity tag before proceeding. Of course, this introduces an additional round-trip, and a timing window during which overlapping updates by other clients would go unnoticed.¶
Below we repeat the example from above (Appendix A.2), but here the origin server returns entity tags for all write operations, and the user agent knows how to deal with them. That is, both take advantage of [RFC2616] already allows.¶
>> Request (3)
PROPPATCH /test.html HTTP/1.1 Host: example.com Content-Type: application/xml <proppatch xmlns="DAV:"> <set> <prop> <title xmlns="http://ns.example.org/"> A sample document </title> </prop> </set> </proppatch>
>> Response (3)
HTTP/1.1 207 Multi-Status Content-Type: application/xml ETag: "C" <multistatus xmlns="DAV:"> <response> <href>/test.html</href> <propstat> <prop> <title xmlns="http://ns.example.org/"/> </prop> <status>HTTP/1.1 2OO OK</status> </propstat> </response> </multistatus>
As before, this causes the entity to change, and a new entity tag to be assigned. But in this case, the origin server actually notifies the client of the changed state by including the "ETag" response header.¶
The client now will be aware that the requested entity change, and can use the new entity tag in subsequent requests (potentially after refreshing the local copy).¶
Copyright © The Internet Society (2006).¶
This document is subject to the rights, licenses and restrictions contained in BCP 78, and except as set forth therein, the authors retain all their rights.¶
This document and the information contained herein are provided on an “AS IS” basis and THE CONTRIBUTOR, THE ORGANIZATION HE/SHE REPRESENTS OR IS SPONSORED BY (IF ANY), THE INTERNET SOCIETY AND THE INTERNET ENGINEERING TASK FORCE DISCLAIM ALL WARRANTIES, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTY THAT THE USE OF THE INFORMATION HEREIN WILL NOT INFRINGE ANY RIGHTS OR ANY IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.¶
The IETF takes no position regarding the validity or scope of any Intellectual Property Rights or other rights that might be claimed to pertain to the implementation or use of the technology described in this document or the extent to which any license under such rights might or might not be available; nor does it represent that it has made any independent effort to identify any such rights. Information on the procedures with respect to rights in RFC documents can be found in BCP 78 and BCP 79.¶
Copies of IPR disclosures made to the IETF Secretariat and any assurances of licenses to be made available, or the result of an attempt made to obtain a general license or permission for the use of such proprietary rights by implementers or users of this specification can be obtained from the IETF on-line IPR repository at http://www.ietf.org/ipr.¶
The IETF invites any interested party to bring to its attention any copyrights, patents or patent applications, or other proprietary rights that may cover technology that may be required to implement this standard. Please address the information to the IETF at ietf-ipr@ietf.org.¶