如何在给定方案中的WCF RESTful服务中创建方法?

时间:2015-07-30 10:39:48

标签: c# wcf rest wcf-rest

我必须创建一个PUT RESTful WCF服务,它将具有如下所示的URI模板:

/rs/close_copy/{user_token}?term={term}&brand={brandname}

发给我们的请求有一个JSON主体,格式为:

“acc”: 
“counters”:[ 
{“format”: 
“ink”: 
“ctr”:
“duplex”: },
{…}]

但问题是,上面的"counters"参数并不总是像预期的那样作为JSON对象的数组出现。当"counter"中只有一个元素时,请求作为单个JSON对象出现,而不是带有一个元素的JSON对象列表。

是第三方致电我们的服务而他们无法对其请求进行更改。我在WCF中实现了如下内容:

[WebInvoke(Method = "PUT", ResponseFormat = WebMessageFormat.Json,
           RequestFormat = WebMessageFormat.Json, BodyStyle = WebMessageBodyStyle.Bare,
           UriTemplate = "/rs/close_copy/{user_token}?term={term}&brand={brandname}")]
JsonResponse EndSession(string user_token, string term, string brandname, EndSessionRequest request);

其中EndSessionRequest是:

[DataContract]
public class EndSessionRequest
{
    [DataMember]
    public string acc { get; set; }        

    [DataMember]
    public IEnumerable<PageDetails> counters { get; set; }
}

PageDetails是:

[DataContract]
public class PageDetails
{
    [DataMember]
    public string format { get; set; }
    [DataMember]
    public string ink { get; set; }
    [DataMember]
    public int ctr { get; set; }
    [DataMember]
    public bool duplex { get; set; }
}

上述实现的问题是,当counters有一个元素时,请求就像:

{"acc":"ramaccnz","counters":{"ctr":"2","duplex":"false","format":"A4","ink":"bw"}}

但根据我们的实施,请求预期为:

{"acc":"ramaccnz","counters":[{"ctr":"2","duplex":"false","format":"A4","ink":"bw"}]} 

在其他情况下,当请求包含多个元素时,我们的服务会按预期工作。

有没有办法在WCF实现中处理这个问题?

2 个答案:

答案 0 :(得分:1)

好的。这有点棘手。我目前正在开发RESTful服务。有一种称为CollectionDataContract的东西也许是实现目标的一种方式,但我是WCF的新手,所以我不能告诉你更多。 另一种方法是获取您请求的原始数据流,因此您必须使用JavaScriptSerializer自己获取和反序列化JSON数据。在这种情况下,您可以自己处理输入数据。像..

 public JsonResponse ServiceCall(Stream dataStream, object parameter) {
     var dataBytes = dataStream.ReadToEnd();
     // use the required encoding to get the string data
     var dataString = Encoding.UTF8.GetString(dataBytes);
     var dataJson = default(EndSessionRequest);
     try { dataJson = new JavaScriptSerializer().Deserialize<EndSessionRequest>(); }
     catch {  
         // the request includes just one entry that's why
         // the serializer fails getting the object so
         // you could continue like..
         dataJson = new JavaScriptSerializer().Deserialize<EndSessionRequestWithSingleCounter>();
     }
     // handle request ...
 }

要在请求调用方法中使用原始流,只需删除参数类型规范 EndSessionRequest 并插入。您是 Web.config 在这种情况下还需要进行一些更改..

 <!-- insert into system.web tag -->
 <system.web>
     <httpRuntime targetFramework="4.5" maxRequestLength="2000000" />
     <!-- other stuff.. -->
 </system.web>
 <!-- insert into system.serviceModel > bindings -->
 <system.serviceModel>
     <bindings>
         <webHttpBinding>
             <binding 
                 name="YourBindingName"
                 maxBufferSize="65536"
                 maxreceiveMessageSize="2000000000"
                 transferMode="Streamed" />
         </webHttpBinding>
     </bindings>
 </system.serviceModel>

不要忘记将 bindingConfiguration =“YourBindingName” 设置为您的服务行为。

但正如 @Ricardo Pontual 已经提到的那样。这是第三方以错误的方式调用您的服务..

答案 1 :(得分:0)

我能够通过使用OperationContext以JSON格式获取请求字符串,然后使用DataContractJsonSerializer手动处理请求来完成此操作。以下是我的代码段:

[DataContract]
public class EndSessionRequestMany
{
    [DataMember]
    public string acc { get; set; }
    [DataMember]
    public List<PageDetails> counters { get; set; }
}

public JsonResponse EndSession(string user_token, string term, string brandname)
{
    string JSONstring = OperationContext.Current.RequestContext.RequestMessage.ToString();

    XmlReader reader = XmlReader.Create(new StringReader(JSONstring));
    EndSessionRequestMany objEndSessionMany = (EndSessionRequestMany)new DataContractJsonSerializer(typeof(EndSessionRequestMany)).ReadObject(reader);
}