1

我故意尝试使用 Newtonsoft Json 创建无效的 JSON,以便放置一个 ESI 包含标签,这将获取另外两个 json 节点。

这是我的 JsonConverter 的 WriteJson 方法:

public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
    mApiResponseClass objectFromApi = (mApiResponseClass)value;

    foreach (var obj in objectFromApi.GetType().GetProperties())
    {
        if (obj.Name == "EsiObj")
        {
            writer.WriteRawValue(objectFromApi.EsiObj);
        }
        else
        {
            writer.WritePropertyName(obj.Name);
            serializer.Serialize(writer, obj.GetValue(value, null));
        }
    }
}

mApiResponseClass 中的 EsiObj 只是一个字符串,但需要将其写入 JSON 响应以在没有任何属性名称的情况下进行解释 - 以便 hte ESI 可以工作。

这当然会导致 Json Writer 出现异常,其值为:

Newtonsoft.Json.JsonWriterException:'状态对象中未定义的令牌将导致无效的 JSON 对象。小路 ''。'

有没有办法解决?

一个理想的输出应该是 JSON 格式的,技术上是无效的,看起来像这样:

{
value:7,
string1:"woohoo",
<esi:include src="/something" />
Song:["I am a small API","all i do is run","but from who?","nobody knows"]
}

编辑: 使用 ESI 允许我们对单个响应具有不同的缓存长度 - 即我们可以将可以缓存很长时间的数据放在 JSON 的某些部分中,并且只获取更新的部分,例如那些依赖于客户端的部分- 特定数据。ESI 不是特定于 HTML 的。(如下文所示)它通过支持这些标签的 Varnish 运行。不幸的是,我们要求我们只发布 1 个文件作为响应,并且不需要客户的进一步请求。我们也不能改变我们的响应——所以我不能只添加一个 JSON 节点来专门包含其他节点。

编辑 2: “更多 json 节点”部分通过 ESI 向我们的后端进一步请求用户/客户端特定数据(即向另一个端点)解决。预期的结果是我们随后将原始 JSON 文档和后来请求的文档无缝地合并在一起。(这样,原始文档可以是旧的,而客户特定的可以是新的)

编辑 3: 端点 /something 将输出类似 JSON 的片段,例如:

teapots:[ {Id: 1, WaterLevel: 100, Temperature: 74, ShortAndStout: true}, {Id: 2, WaterLevel: 47, Temperature: 32, ShortAndStout: true} ],

对于以下的总响应:

{
value:7,
string1:"woohoo",
teapots:[ {Id: 1, WaterLevel: 100, Temperature: 74, ShortAndStout: true}, {Id: 2, WaterLevel: 47, Temperature: 32, ShortAndStout: true} ],
Song:["I am a small API","all i do is run","but from who?","nobody knows"]
}
4

1 回答 1

5

您的基本问题是 aJsonWriter是一个状态机,它跟踪当前JSON 状态并验证从状态到状态的转换,从而确保不编写结构不良的 JSON。这是以两种不同的方式绊倒你。

首先,您的WriteJson()方法不是调用WriteStartObject()and WriteEndObject()。这些是编写{}围绕 JSON 对象的方法。由于您的“理想输出”显示了这些大括号,因此您应该在WriteJson().

其次,您WriteRawValue()在格式良好的 JSON 不允许出现值的地方调用,特别是在需要属性名称的地方。预计这会导致异常,因为文档指出:

在需要值的地方写入原始 JSON 并更新写入器的状态。

您可以改为使用的WriteRaw()记录如下:

写入原始 JSON 而不更改编写器的状态。

但是,WriteRaw()不会给你任何好处。具体来说,您需要自己编写任何分隔符和缩进。

解决方法是将您的转换器修改为如下所示:

public class EsiObjConverter<T> : JsonConverter
{
    const string EsiObjName = "EsiObj";

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        var contract = serializer.ContractResolver.ResolveContract(value.GetType()) as JsonObjectContract;
        if (contract == null)
            throw new JsonSerializationException(string.Format("Non-object type {0}", value));
        writer.WriteStartObject();
        int propertyCount = 0;
        bool lastWasEsiProperty = false;
        foreach (var property in contract.Properties.Where(p => p.Readable && !p.Ignored))
        {
            if (property.UnderlyingName == EsiObjName && property.PropertyType == typeof(string))
            {
                var esiValue = (string)property.ValueProvider.GetValue(value);
                if (!string.IsNullOrEmpty(esiValue))
                {
                    if (propertyCount > 0)
                    {
                        WriteValueDelimiter(writer);
                    }
                    writer.WriteWhitespace("\n");
                    writer.WriteRaw(esiValue);
                    // If it makes replacement easier, you could force the ESI string to be on its own line by calling
                    // writer.WriteWhitespace("\n");

                    propertyCount++;
                    lastWasEsiProperty = true;
                }
            }
            else
            {
                var propertyValue = property.ValueProvider.GetValue(value);

                // Here you might check NullValueHandling, ShouldSerialize(), ...

                if (propertyCount == 1 && lastWasEsiProperty)
                {
                    WriteValueDelimiter(writer);
                }
                writer.WritePropertyName(property.PropertyName);
                serializer.Serialize(writer, propertyValue);

                propertyCount++;
                lastWasEsiProperty = false;
            }
        }
        writer.WriteEndObject();
    }

    static void WriteValueDelimiter(JsonWriter writer)
    {
        var args = new object[0];
        // protected virtual void WriteValueDelimiter() 
        // https://www.newtonsoft.com/json/help/html/M_Newtonsoft_Json_JsonWriter_WriteValueDelimiter.htm
        // Since this is overridable by client code it is unlikely to be removed.
        writer.GetType().GetMethod("WriteValueDelimiter", BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic).Invoke(writer, args);
    }

    public override bool CanConvert(Type objectType)
    {
        return typeof(T).IsAssignableFrom(objectType);
    }

    public override bool CanRead { get { return false; } }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        throw new NotImplementedException();
    }
}

序列化的输出将是:

{
  "value": 7,
  "string1": "woohoo",
<esi:include src="/something" />,
  "Song": [
    "I am a small API",
    "all i do is run",
    "but from who?",
    "nobody knows"
  ]
}

现在,在您的问题中,您想要的 JSON 输出显示未正确引用的 JSON 属性名称。如果您真的需要这个并且它不仅仅是问题中的错字,您可以通过设置JsonTextWriter.QuoteName为来完成此操作,false如对Json.Net - Serialize property name without quotes by Christophe Geers的回答中所示:

var settings = new JsonSerializerSettings
{
    Converters = { new EsiObjConverter<mApiResponseClass>() },
};    
var stringWriter = new StringWriter();
using (var writer = new JsonTextWriter(stringWriter))
{
    writer.QuoteName = false;
    writer.Formatting = Formatting.Indented;
    writer.Indentation = 0;
    JsonSerializer.CreateDefault(settings).Serialize(writer, obj);
}

结果是:

{
value: 7,
string1: "woohoo",
<esi:include src="/something" />,
Song: [
"I am a small API",
"all i do is run",
"but from who?",
"nobody knows"
]
}

几乎是您的问题中显示的内容,但不完全是。它在 ESI 字符串和下一个属性之间包含一个逗号分隔符,但在您的问题中没有分隔符:

<esi:include src="/something" /> Song: [ ... ]

摆脱分隔符被证明是难以实现的,因为JsonTextWriter.WritePropertyName()当不在对象的开头时会自动写入分隔符。不过,我认为这应该是可以接受的。ESI 本身不知道它是在替换对象的 first、last 还是 middle 属性,因此最好不要在替换字符串中包含分隔符。

工作示例 .Net在这里摆弄。

于 2018-11-13T19:22:51.827 回答