解析JSON、扩展Fiddler
按文章结构,这部分应该给出WCFRest项目示例,我想WinForm示例足够详尽了,况且WCFRest还不需要使用插件AppDomain那一套,于是把最近写的Fiddler扩展搬上来吧。
Fiddler有一套自成的插件系统,可以在其官方网站找到完整文档(戳这里)。通过其提供的一整套接口,我们可以从界面至功能全方位扩展它。这里主题简单,我们只为其添加一个JSON解析界面。
PART I:JSON解析
Mgen有一个JSON解析范例(戳这里)代码相当好看,WPF模块绑定也很强大。这里使用Json.com的一个示例稍作修改,解析效果如下:
上述代码过于忠实地体现了Newtonsoft的设计,只有JValue被解析成叶节点,其他JToken对象全部是枝节点,解析出来的JSON树层次太深可读性不够好。
Newtonsoft中JSON结构:
这里重新设计结构图如下:
一共有4种节点:指示值的JsonValueNode叶节点、指示键值对的JsonPropertyNode的叶节点,指示数组的JsonArrayNode枝节点,及指示对象的JsonObjectNode的树节点。解析时关键在对JProperty的处理:把Value为JValue的JProperty对象解析成JsonPropertyNode叶节点,Value为JContainer的JProperty视子节点属性实例成JsonArrayNode或JsonObjectNode树节点,初步代码如下:
public class JsonNode
{
public IEnumerable<JsonNode> Children { get; internal set; } internal JsonNode()
{
}
} public class JsonValueNode : JsonNode
{
public Object Value { get; private set; } public JsonValueNode(Object value)
{
Value = value;
} public override String ToString()
{
return Value != null ? Value.ToString() : "<null>";
}
} public class JsonPropertyNode : JsonNode
{
public String Name { get; private set; }
public Object Value { get; private set; } public JsonPropertyNode(String name, Object value)
{
Name = name;
Value = value;
} public override String ToString()
{
return String.Concat(Name, " : ", (Value != null ? Value.ToString() : "<null>"));
}
} public class JsonArrayNode : JsonNode
{
public String Name { get; private set; } public JsonArrayNode(String name)
{
Name = name ?? "[]";
} public override String ToString()
{
return Name;
}
} public class JsonObjectNode : JsonNode
{
public String Name { get; private set; } public JsonObjectNode(String name)
{
Name = name ?? "{}";
} public override String ToString()
{
return Name;
}
}
JsonNodeFactory作为JsonNode的建造者:
public class JsonNodeFactory
{
public static JsonNode CreateFromJToken(JToken jtoken)
{
if (jtoken is JValue)
{
return new JsonValueNode(((JValue)jtoken).Value);
}
else if (jtoken is JProperty)
{
JProperty jproperty = (JProperty)jtoken;
if (jproperty.Value is JValue)
{
return new JsonPropertyNode(jproperty.Name, ((JValue)jproperty.Value).Value);
}
else if (jproperty.Value is JArray)
{
JsonArrayNode jsonNode = new JsonArrayNode(jproperty.Name);
jsonNode.Children = ((JArray)jproperty.Value).Children().Select(n => CreateFromJToken(n));
return jsonNode;
}
else if (jproperty.Value is JObject)
{
JsonObjectNode jsonNode = new JsonObjectNode(jproperty.Name);
jsonNode.Children = ((JObject)jproperty.Value).Children().Select(n => CreateFromJToken(n));
return jsonNode;
}
else
{
throw new Exception("Unknown JProperty");
}
}
else if (jtoken is JArray)
{
JsonArrayNode jsonNode = new JsonArrayNode(null);
jsonNode.Children = ((JArray)jtoken).Children().Select(n => CreateFromJToken(n));
return jsonNode;
}
else if (jtoken is JObject)
{
JsonObjectNode jsonNode = new JsonObjectNode(null);
jsonNode.Children = ((JObject)jtoken).Children().Select(n => CreateFromJToken(n));
return jsonNode;
}
else
{
throw new Exception("Unknown jtoken");
}
}
}
文章后面的代码文件中更具体的实现加入了父节点与索引,ToString()逻辑更完备。客户端调用:
class Program
{
static void Main(string[] args)
{
JToken jtoken = JToken.Parse(System.IO.File.ReadAllText("json.txt"));
JsonNode node = JsonNodeFactory.CreateFromJToken(jtoken);
Debug.Listeners.Add(new TextWriterTraceListener(Console.Out));
Display(node);
} private static void Display(JsonNode node)
{
Debug.WriteLine(node);
if (node.Children != null)
{
Debug.Indent();
foreach (JsonNode sub in node.Children)
{
Display(sub);
}
Debug.Unindent();
}
}
}
json.txt见截图与后文代码文件,新的解析结构:
PART2:Fiddler插件
Fiddler自带的JSON显示是一个简单的Tree,无法完成复杂功能。Codeplex上有一个JsonView项目(戳这里),Fiddler子项目丢到%Program Files%/Fiddler/Inspectors目录即可。问题在于它有BUG,且使用Newtonsoft的Json.Net版本极其低。没办法,重写一个。这里有2处需要注意:
1. 需要添加Public类,实现Inspector2、IResponseInspector2,抽象类Inspector2.AddToTab(TabPage o)是UI呈现方法,headers与body属性内部可以完成对自定义控件赋值;
2. 需要加入[assembly: Fiddler.RequiredVersion("x.x.x.x")]特性,位置不限。Fiddler目录有基于.Net Framework 2.0和4.0的版本,本例使用4.0,CLR版本兼容性、X64兼容性等具体内容请自行翻阅文档。
AddToTab方法大致内容如下:
public override void AddToTab(TabPage tabPage)
{
utrlJson = new UserControl_JsonView() { Dock = DockStyle.Fill };
tabPage.Text = "MyJson";
tabPage.Controls.Add(utrlJson);
}
这里使用了一个UserControl,暴露一个Content属性,内部使用TextBox和TreeView展示Json文本与JsonNode结构。
取消Fiddler引用及排除Plugin.cs,将项目类型设置为Windows应用程序,可以得到Form窗体程序;加入引用及Plugin.cs,将项目类型设置成类库,扔到%Program Files%/Fiddler/Inspectors目录便是Fiddler插件。这里加入了文本定位、节点分层展开、节点值复制(可以在TreeView上使用Ctrl+C进行智能复制)等方法,方便使用。
PART3:后记
感觉代码还是比较乱,JsonNode创建方法还可以提炼;水平所限,TextBox光标定位正则不够强大,需要优化;功能上讲,可以加入设置项以组织编码格式、Header解析、控件视图等。代码文件及二进制文件(戳这里)奉上。