zoukankan      html  css  js  c++  java
  • 动态加载与插件系统的初步实现(四):解析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解析、控件视图等。代码文件及二进制文件(戳这里)奉上。

    附求职信息:目前在北京,寻求.Net相关职位,偏向后端,请邮件jusfr.v#gmail.com,替换#为@,沟通后奉上简历。

    更新内容:

    • 7/2:添加了Tree递归与回溯方法以进行关键字查找染色;修复了Clipboard.SetText()方法未进行空白值检查的BUG;
    • 7/3:完善关键字查找与层级展开逻辑的互相影响,显示更合理;
  • 相关阅读:
    window+Apache+php+mysql注意事项
    【转】QRCode二维码的应用心得
    如何创建和配置服务器证书进行SSL Relay
    Citrix 实践中的问题及解决
    web.config中httpRunTime的属性
    [原]IE9 DOM的自定义属性问题
    添加图片到数据库
    Windows server 2008安装企业CA证书服务
    EXTJS Date 的转换格式化字符
    Extjs 4.1 学习笔记(一)(proxy,loader,treestore)
  • 原文地址:https://www.cnblogs.com/Jusfr/p/3167080.html
Copyright © 2011-2022 走看看