WCF REST - " Get"使用复杂的类型

时间:2013-07-02 22:09:26

标签: c# wcf wcf-rest

我有一个WCF 4 RESTful Web服务,它必须在GET(它的JSONP)上使用。我为此服务添加了一个方法,它需要一个复杂的类型参数。

[OperationContract]
[WebGet(RequestFormat=WebMessageFormat.Json)]
void InsertEventActionList(string pageViewGUID, 
                             List<EventActionItem> eventActionItemList);

然后我收到错误:

Operation 'InsertEventActionList' in contract 'ITrackingService' 
has a query variable named 'eventActionItemList' of type 
'System.Collections.Generic.List`1[EventActionItem]', but type 
'System.Collections.Generic.List`1[EventActionItem]' is not convertible by 
'QueryStringConverter'.  Variables for UriTemplate query values 
must have types that can be converted by 'QueryStringConverter'.

好的,所以使用这篇文章 - http://blogs.msdn.com/b/carlosfigueira/archive/2011/08/09/wcf-extensibility-querystringconverter.aspx我创建了一个附加到行为扩展的自定义QueryStringConverter:

[DataContract]
public class EventActionItem
{
    [DataMember]
    public string ActionName { get; set; }

    [DataMember]
    public string ActionValue { get; set; }

    [DataMember]
    public string ActionCategory { get; set; }
}

public class EventActionItemQueryStringConverter : QueryStringConverter
{
    public override bool CanConvert(Type type)
    {
        return type == typeof(List<EventActionItem>) || base.CanConvert(type);
    }

    public override object ConvertStringToValue(string parameter, Type parameterType)
    {
        if (parameterType == typeof(List<EventActionItem>))
        {
            List<EventActionItem> item = JsonConvert.DeserializeObject<List<EventActionItem>>(parameter);

            if (item != default(List<EventActionItem>))
            {
                return item;
            }
        }

        return base.ConvertStringToValue(parameter, parameterType);
    }
}

public class TrackingWebHttpBehavior : WebHttpBehavior
{
    protected override QueryStringConverter GetQueryStringConverter(OperationDescription operationDescription)
    {
        return new EventActionItemQueryStringConverter();
    }
}

public class TrackingBehaviorExtension : BehaviorExtensionElement
{
    public override Type BehaviorType
    {
        get { return typeof(TrackingWebHttpBehavior); }
    }

    protected override object CreateBehavior()
    {
        return new TrackingWebHttpBehavior();
    }
}

的web.config:

<system.serviceModel>
    <extensions>
      <behaviorExtensions>
        <add name="queryString"
             type="namespace.TrackingBehaviorExtension, namespace"/>
      </behaviorExtensions>
    </extensions>
    <behaviors>
      <serviceBehaviors>
        <behavior>
          <serviceMetadata httpGetEnabled="true"/>
          <serviceDebug includeExceptionDetailInFaults="false"/>
        </behavior>
      </serviceBehaviors>
      <endpointBehaviors>
        <behavior>
          <queryString />
        </behavior>
      </endpointBehaviors>
    </behaviors>
    <serviceHostingEnvironment aspNetCompatibilityEnabled="true" multipleSiteBindingsEnabled="true"/>
    <standardEndpoints>
      <webHttpEndpoint>
        <standardEndpoint name="" helpEnabled="true" automaticFormatSelectionEnabled="true" />
      </webHttpEndpoint>
    </standardEndpoints>
  </system.serviceModel>

但是,我仍然遇到同样的错误。我已经调试了服务初始化并且正在创建BehaviorExtensionElement,但永远不会调用QueryStringConverter

是因为我使用standardEndpoint配置而不是客户用户定义的端点吗?如果是,我如何将自定义行为附加到standardEndpoint

1 个答案:

答案 0 :(得分:0)

修改

根据您正在开发的内容,您可以使用https://github.com/oyvindkinsey/easyXDM 它允许在x域中进行POST(使用XHR)。


查询字符串不是太多了吗?我想如果它只是一个需要传递的对象,那么查询字符串可以正常,但是列表......?也许我错了..但我认为你应该只为每个请求传递“一个”EventActionItem。

如果你能负担得起这些变化,只需将列表(或数组)作为一个body参数,然后使用POST代替GET,或者像我上面建议的那样,对EventActionItem数组中的每个项目提出一个请求。