前一节介绍了一种IETF推荐的一种超媒体格式------Atom,这一节中主要Atom发布协议-------Atom Publish Protocol,简称AtomPub,有时更简洁写作APP。
开篇之前介绍几个重要概念先:
1、媒体类型
描述相关资源表述所使用的类型,如XML、JSON、JPG、MP3等、处理模型以及链接关系值
2、HTTP惯用语
它规范了如何对资源进行操作以及处理HTTP头信息和状态码
3、领域应用协议。
领域应用协议(Domain Application Protocol DAP),它为服务端与客户单的资源的交互定义了一套规范,如资源的媒体类型、链接关系值、以及HTTP惯用语。DAP居于HTTP之上,具体化了HTTP的广义应用。服务实现DAP后,通过在返回给客户端的资源的表述中添加链接,以对资源进行广告,提示消费者可以进行哪些操作来驱动业务的运行。
4、条目资源(entry resource):可以使用atom条目表现的成员,如文本
5、媒体资源(media resource):不可以使用atom条目表现的成员,如视频、音乐等可执行文件
6、代理资源(proxy resource):没有资源连接条目,被插入到集合中,用来替代媒体资源。
AtomPub是一种建立在Atom超媒体格式之上的应用协议,用于发布和编辑Web资源。它采用一种标准化的机制来创建和编辑Web资源,并能解决任何可能发生的冲突,如对Web资源操作时的并发等问题。本节中将说明Atom如何实现服务端与客户端如何实施一套领域应用协议(Doamin Application Protocol ,简称DAP)。
目录:
1、AtomPub概述
2、AtomPub四个要素
3、AtomPub的扩展
4、AtomPub如何解决并发操作
5、REST WCF中使用AtomPub
6、.Net中消费Atom资源
1、AtomPub概述:
1.1、AtomPub是一个应用协议,而不是传输协议,制定了服务和消费者对资源操作的规范,所以它是基于对资源的发布和编辑设计的
1.2、AtomPub指明服务发回给消费者的Atom资源之间如何进行转移
AtomPub处理模型规范了一下几点:1、资源以何种格式进行表述 2、消费者操作资源的HTTP惯用语 3、服务端如何通对资源进行广告 4、超媒体控件的标记
2、AtomPub四个要素。
成员和集合是针对发布活动的事物的抽象名词。成员是封装了资源的表述或处于草案状态等待发布资源的表述。集合是一组成员。如下为一个包含三个成员的集合.服务文档与分类文档见下图:
图1
图2
2.1、集合
集合是在服务文档中定义的。在实际开发过程中,点击图2中的collection 后的href,就应该能获取到一个类似图1的集合。消费者通过每个集合的href属性与每个集合所包括的资源交互。在图1中,此集合就是一个包含三个条目(entry)的提要(feed)的Atom表述。AtomPub中没有规定如何创建或者删除集合。集合支持一下操作:
集合通过Accept指定资源所支持的媒体类型。集合通过分类文档对多种类别进行广告。
2.2、成员
对成员的操作与集合类似。注意:对于媒体资源,应当使用代理资源进行处理,也就是在集合插入代理资源,然后通过制定的链接来获取媒体资源。
2.3、分类文档
分类文档包含对集合和成语进行分类的列表。如下图:
2.4、服务文档
服务文档包含访问集合而对外公布的入口点。图2就是一个服务文档。
集合与成员描述了Web资源的表述形式,分类文档与服务文档则从整体上描述协议。一个服务文档,可以包含多个工作区,它通过工作区来对集合进行分组;一个集合可以出现在多个工作区中。而一个工作区又包括多个成员。
3、AtomPub的扩展
3.1、对Atom的扩展
SyndicationItem item = new SyndicationItem("item4", "content4", new Uri("http://tyb1222.cnblogs.com/", UriKind.Absolute), "Id", DateTime.Now); item.ElementExtensions.Add(new SyndicationElementExtension("control","tyb1222.cnblogs.com","control content"));
edited元素表示一个成员创建时间或者编辑。集合中的成员通过edited倒序排列,它总是由服务区控制的。control元素用于发布控件。发布控件是致力于控制发布生命周期的Atom扩展元素。<draft>就是一个发布控件,它的值表示客户对成员可见性的期望。
3.2、对HTTP头进行扩展
AtomPub引入了一个新的HTTP头信息:Slug。当客户端提交一个新成员时,可以包含一个Slug头信息。这个Slug头信息代表了对服务器的一个请求:将这个头信息放入资源表述的ID、标题或者URI中。客户端通过它来让服务端创建人类可读的URI。
4、AtomPub如何解决并发操作
通过edit链接可以获取一个资源的表述,这样就可能导致多个不同的消费者并发操作改资源。在这种情况下,一个消费者对资源所做的操作可能被另外一个消费者所覆盖,这就是所谓的丢失更新。为了解决丢失更新,服务端可以选择将资源挂起,直到资源一个生命周期的结束,这是一种悲观锁的方式来解决这个问题。但是它同时限制了在资源生命周期中对资源的除了GET之外的其他操作。AtomPub也可以通过之前介绍的ETag(实体标签:EntityTag)和条件匹配(If-Match)通过乐观锁定的方式来解决这个问题。
1、客户端通过POST请求创建资源,创建成功后,服务端返回:
HTTP/1.1 201 Created
Location: Order/3ad3b19b-a7fc-4ee6-a446-fa4b2d9b0140
ETag: "634600748498750000"
2、此时不同的消费者都可以通过Location指定的值获取资源的表述。当不同的消费者发送修改资源的请求,如PUT\DELETE时,需要向请求HTTP头中添加If-Match信息与ETag值。
如:
PUT Order/3ad3b19b-a7fc-4ee6-a446-fa4b2d9b0140 ...... If-Match:"634600748498750000"
如果自从最近一次生成ETag值以来发生过变化,则服务器返回412状态码,即 ttpStatusCode.PreconditionFailed。当消费者获取到412状态码后,它必须决定下一步业务如何进行。最常见的方式是再次发送GET请求以获取最新的资源表述以及ETag值,然后连同ETag值再次通过条件匹配发送到服务端。
关于ETag值以及并发控制,之后还会有更详细的介绍,敬请期待。。。
5、REST WCF中使用AtomPub
使用WCF构建REST架构风格的服务时,在.Net FCL(Framework Class Library)中使用AtomPub最基本的是如何发布AtomPub的服务文档。FCL中封装了一些 类来为我们发布服务文档,如:AtomPub10ServiceDocumentFormatter
ResourceCollectionInfo、Workspace、CategoryDocument、ServiceDocument。代码如下:
List<ResourceCollectionInfo> resourceCollectionInfos = new List<ResourceCollectionInfo>(); ResourceCollectionInfo resourceCollectionInfo = new ResourceCollectionInfo("Title", new Uri("tyb1222", UriKind.Relative)); CategoriesDocument categoriesDocument = CategoriesDocument.Create(new Uri("http://www.cnblogs.com/tyb1222", UriKind.Absolute)); categoriesDocument.ElementExtensions.Add(new SyndicationElementExtension("outer", "outNamespace", 1000)); categoriesDocument.ElementExtensions.Add(new SyndicationElementExtension("in", "outNamespace", "inValue")); resourceCollectionInfo.Categories.Add(categoriesDocument); resourceCollectionInfo.Accepts.Add("application/atom+xml;type=entry"); resourceCollectionInfos.Add(resourceCollectionInfo); ResourceCollectionInfo resourceCollectionInfo2 = new ResourceCollectionInfo("Title2", new Uri("tyb1222", UriKind.Relative)); resourceCollectionInfo2.Accepts.Add("image/png"); resourceCollectionInfos.Add(resourceCollectionInfo2); List<Workspace> workspaces = new List<Workspace> { new Workspace("workspace1",resourceCollectionInfos), }; ServiceDocument serviceDocument = new ServiceDocument(workspaces) { BaseUri = new Uri("http://localhost:1209/AtomPub/feed") }; AtomPub10ServiceDocumentFormatter serviceDocumentFormatter = serviceDocument.GetFormatter() as AtomPub10ServiceDocumentFormatter;
运行服务代码如上图2。
当点击<app:collection href="tyb1222">中的href链接,则能获取到此集合资源,以超媒体格式Atom显示。如上图1所示。
6、.Net中消费Atom资源
由于Atom也是一种XML格式,所以在.Net中,直接通过获取资源的URI读取到XML文件内容,然后再对资源做相应处理
static void ConsumeAtom() { XmlReader reader = XmlReader.Create("http://localhost:1209/AtomPub/feed/tyb1222"); SyndicationFeed feed = SyndicationFeed.Load(reader); List<SyndicationItem> syndicationItems = feed.Items.ToList(); syndicationItems.ForEach(item=> { Console.WriteLine(item.Title.Text); }); }
运行结果如下图: