6

我正在使用 flurl 提交 HTTP 请求,这非常有用。现在我需要将一些请求的“ Content-Type”标头更改为“application/json;odata=verbose”

    public async Task<Job> AddJob()
    {

        var flurlClient = GetBaseUrlForGetOperations("Jobs").WithHeader("Content-Type", "application/json;odata=verbose");
        return await flurlClient.PostJsonAsync(new
        {
            //Some parameters here which are not the problem since tested with Postman

        }).ReceiveJson<Job>();
    }

    private IFlurlClient GetBaseUrlForOperations(string resource)
    {
        var url = _azureApiUrl
            .AppendPathSegment("api")
            .AppendPathSegment(resource)
            .WithOAuthBearerToken(AzureAuthentication.AccessToken)
            .WithHeader("x-ms-version", "2.11")
            .WithHeader("Accept", "application/json");
        return url;
    }

您可以看到我如何尝试在上面添加标题 ( .WithHeader("Content-Type", "application/json;odata=verbose"))

不幸的是,这给了我以下错误:

“InvalidOperationException:错误的标头名称。确保请求标头与 HttpRequestMessage 一起使用,响应标头与 HttpResponseMessage 一起使用,内容标头与 HttpContent 对象一起使用。”

我还尝试了 flurl 的“ConfigureHttpClient”方法,但找不到设置内容类型标头的方式/位置。

4

5 回答 5

5

这个答案已经过时了。升级到最新版本(2.0 或更高版本),问题就消失了。

事实证明,真正的问题System.Net.HttpAPI 如何验证标头有关。它区分了请求级标头和内容级标头,我一直觉得这有点奇怪,因为原始 HTTP 没有这种区别(可能在多部分场景中除外)。FlurlWithHeader将标头添加到HttpRequestMessage对象,但未通过验证Content-Type,它希望将其添加到HttpContent对象中。

这些 API 确实允许您跳过验证,虽然 Flurl 没有直接公开它,但您可以很容易地进入底层,而不会破坏 fluent 链:

return await GetBaseUrlForGetOperations("Jobs")
    .ConfigureHttpClient(c => c.DefaultRequestHeaders.TryAddWithoutValidation("Content-Type", "application/json;odata=verbose"))
    .PostJsonAsync(new { ... })
    .ReceiveJson<Job>();

这可能是做你需要的最好的方法,并且仍然利用 Flurl 的优点,即不必直接处理序列化、HttpContent对象等。

我正在强烈考虑根据这个问题更改 Flurl 的AddHeader(s)实现以使用。TryAddWithoutValidation

于 2017-06-14T15:08:54.017 回答
2

我发现的评论和另一篇文章(当我再次找到它时会添加参考)为我指明了正确的方向。我的问题的解决方案如下:

        var jobInJson = JsonConvert.SerializeObject(job);
        var json = new StringContent(jobInJson, Encoding.UTF8);
        json.Headers.ContentType = MediaTypeHeaderValue.Parse("application/json; odata=verbose");

        var flurClient = GetBaseUrlForOperations("Jobs");

        return await flurClient.PostAsync(json).ReceiveJson<Job>();

编辑:找到相关的 SO 问题:Azure encoding job via REST Fails

于 2017-06-14T11:06:13.640 回答
1

我可以发布同一个问题的 3 个答案吗?:)

升级。Flurl.Http 2.0 包括以下对标头的增强:

  1. WithHeader(s)现在TryAddWithoutValidation在引擎盖下使用。仅通过该更改,OP 的代码将按最初发布的方式工作。

  2. 标头现在设置在请求级别,这解决了另一个已知问题

  3. 使用SetHeaders对象表示法时,属性名称中的下划线将转换为标题名称中的连字符,因为标题中的连字符很常见,下划线不是,C# 标识符中不允许使用连字符。

这对您的情况很有用:

.WithHeaders(new {
    x_ms_version = "2.11",
    Accept = "application/json"
});
于 2017-09-22T17:14:46.460 回答
1
public static class Utils
{
    public static IFlurlClient GetBaseUrlForOperations(string resource)
    {
        var _apiUrl = "https://api.mobile.azure.com/v0.1/apps/";

        var url = _apiUrl
            .AppendPathSegment("Red-Space")
            .AppendPathSegment("HD")
            .AppendPathSegment("push")
            .AppendPathSegment("notifications")
            .WithHeader("Accept", "application/json")
            .WithHeader("X-API-Token", "myapitocken");

            return url;
    }

    public static async Task Invia()
    {
        FlurlClient _client;
        PushMessage pushMessage = new PushMessage();
        pushMessage.notification_content = new NotificationContent();

        try
        {
            var flurClient = Utils.GetBaseUrlForOperations("risorsa");
            // News news = (News)contentService.GetById(node.Id);
            //pushMessage.notification_target.type = "";
            pushMessage.notification_content.name = "A2";
            // pushMessage.notification_content.title = node.GetValue("TitoloNews").ToString();
            pushMessage.notification_content.title = "Titolo";
            pushMessage.notification_content.body = "Contenuto";
            var jobInJson = JsonConvert.SerializeObject(pushMessage);
            var json = new StringContent(jobInJson, Encoding.UTF8);
            json.Headers.ContentType = MediaTypeHeaderValue.Parse("application/json");
            dynamic data2 = await flurClient.PostAsync(json).ReceiveJson();
            var expandoDic = (IDictionary<string, object>)data2;
            var name = expandoDic["notification_id"];
            Console.WriteLine(name);
        }
        catch (FlurlHttpTimeoutException ex)
        {
            Console.WriteLine(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType + " " + ex);
        }
        catch (FlurlHttpException ex)
        {
            Console.WriteLine(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType + " " + ex);
            if (ex.Call.Response != null)
                Console.WriteLine("Failed with response code " + ex.Call.Response.StatusCode);
            else
                Console.WriteLine("Totally failed before getting a response! " + ex.Message);
        }
        catch (Exception ex)
        {
            Console.WriteLine(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType + " " + ex);
        }
    }
}

public class NotificationTarget
{
    public string type { get; set; }
}

public class CustomData {}

public class NotificationContent
{
    public string name { get; set; }
    public string title { get; set; }
    public string body { get; set; }
    public CustomData custom_data { get; set; }
}

public class PushMessage
{
    public NotificationTarget notification_target { get; set; }
    public NotificationContent notification_content { get; set; }
}
于 2017-09-12T08:25:32.197 回答
0

我不是 OData 专家,也不知道您调用的是什么 API(SharePoint?),但根据我见过的大多数示例,您通常想要做的是要求服务器在响应,而不是在请求中声明您正在发送它。换句话说,您希望在Accept;odata=verbose标头上设置该位,而不是Content-Type。对于 Content-Type 应该足够好,Flurl 会自动为您设置,所以只需尝试这个更改,看看它是否有效:application/json

.WithHeader("Accept", "application/json;odata=verbose");
于 2017-06-13T16:18:11.133 回答