我必须创建一个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实现中处理这个问题?
答案 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);
}