I'm deliberately trying to create invalid JSON with Newtonsoft Json, in order to place an ESI include tag, which will fetch two more json nodes.
This is my JsonConverter's WriteJson method:
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));
}
}
}
The EsiObj in mApiResponseClass is just a string, but it needs to be written into the JSON response to be interpretted without any property name - so that hte ESI can work.
This of course results in an exception with the Json Writer, with value:
Newtonsoft.Json.JsonWriterException: 'Token Undefined in state Object would result in an invalid JSON object. Path ''.'
Is there any way around this?
An ideal output from this would be JSON formatted, technically not valid, and would look like this:
{
value:7,
string1:"woohoo",
<esi:include src="/something" />
Song:["I am a small API","all i do is run","but from who?","nobody knows"]
}
Edit: Using ESI allows us to have varying cache lengths of a single response - i.e. we can place data that can be cached for a very long time in some parts of the JSON, and only fetch updated parts, such as those that rely on client-specific data. ESI is not HTML specific. (As some state below) It's being run via Varnish, which supports these tags. Unfortunately, it's required that we do only put out 1 file as a response, and require no further request from the Client. We cannot alter our response either - so i can't just add a JSON node specifically to contain the other nodes.
Edit 2: The "more json nodes" part is solved by ESI making a further request to our backend for user/client specific data, i.e. to another endpoint. The expected result is that we then merge the original JSON document and the later requested one together seamlessly. (This way, the original document can be old, and client-specific can be new)
Edit 3: The endpoint /something would output JSON-like fragments like:
teapots:[ {Id: 1, WaterLevel: 100, Temperature: 74, ShortAndStout: true}, {Id: 2, WaterLevel: 47, Temperature: 32, ShortAndStout: true} ],
For a total response of:
{
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"]
}
Your basic problem is that a
JsonWriter
is a state machine, tracking the current JSON state and validating transitions from state to state, thereby ensuring that badly structured JSON is not written. This is is tripping you up in two separate ways.Firstly, your
WriteJson()
method is not callingWriteStartObject()
andWriteEndObject()
. These are the methods that write the{
and}
around a JSON object. Since your "ideal output" shows these braces, you should add calls to these methods at the beginning and end of yourWriteJson()
.Secondly, you are calling
WriteRawValue()
at a point where well-formed JSON would not allow a value to occur, specifically where a property name is expected instead. It is expected that this would cause an exception, since the documentation states:What you can instead use is
WriteRaw()
which is documented as follows:However,
WriteRaw()
won't do you any favors. In specific, you will need to take care of writing any delimiters and indentation yourself.The fix would be to modify your converter to look something like:
And the serialized output would be:
Now, in your question, your desired JSON output shows JSON property names that are not properly quoted. If you really need this and it is not just a typo in the question, you can accomplish this by setting
JsonTextWriter.QuoteName
tofalse
as shown in this answer to Json.Net - Serialize property name without quotes by Christophe Geers:Which results in:
This is almost what is shown in your question, but not quite. It includes a comma delimiter between the ESI string and the next property, but in your question there is no delimiter:
Getting rid of the delimiter turns out to be problematic to implement because
JsonTextWriter.WritePropertyName()
automatically writes a delimiter when not at the beginning of an object. I think, however, that this should be acceptable. ESI itself will not know whether it is replacing the first, last or middle property of an object, so it seems best to not include the delimiter in the replacement string at all.Working sample .Net fiddle here.