Hello World
Spiga

使用IronPython检测ASP.NET程序状况(上)

2009-03-23 09:09 by 老赵, 14302 visits

在ASP.NET应用程序运行过程中,很可能会遇到各种意料之外的问题。如果在开发环境下,我们可以设置断点,对程序状态一探究竟。但是很显然,在产品环境中我们几乎无法使用这样的状态。也正因为如此,Dump一个内存快照并进行分析才成为一种“高级技术”,同时在线调试也成为一种需要结合技术能力、分析能力,甚至抗压能力的工作。对于调试和解决问题的探索永远不会停止,各成熟的技术团队几乎都会有一个丰富工具箱,用于应付生产环境中的各种状况。

在维护一些生产环境中的ASP.NET应用程序时,老赵也经常会感到“力不从心”。虽然我们可以建立丰富有效的监控或日志等维护机制,但是调试和分析一次程序经常需要耗费大量的脑细胞。因为我们可以使用的工具大都非常抽象,即使是一个非常微小的问题,也要用较多的时间才能发现“哦,原来是这个变量的值进入了一种奇怪的状态”。如果我们有一种机制,可以直观地检查生产环境中正在运行的程序的状态,那么一定可以大大方便我们的工作。

这篇文章记录的便是老赵的一次探索。

我们的目的是“在程序运行过程中记录状态”。使用ASP.NET的传统机制,例如HttpModule,Trace,往往需要修改配置和部署新的程序集,这样一不小心便会造成应用程序的重启。对于访问量数据量到了一定规模的时候,冷启动的消耗也是相当可观的,不断地冷启动会对应用程序造成非常大的影响。可能我们希望的,只是能够简单地(在不影响现有系统的情况下)执行一段代码,并输出一些内容。要实现这点其实并不难,例如最直接的做法便是提交一段代码,保存成文件,调用CSC将其编译成dll,然后在程序中进行加载。这已经不是一种新鲜的技巧了,它已经用在很多地方,得到了一些不错的效果。

但是,老赵在这里还是想使用IronPython,一个原因是老赵最近也在尝试合理地混用各种语言(F#, IronPython, IronRuby, etc.)编写应用程序以提高生产力,便“顺便”地用上了IronPython的现有成果。IronPython已经内置了Python代码的执行引擎,并且能与.NET程序无缝集成。此外,社区中也已经接受了如Crack.NET等基于IronPython的调试工具1,表明IronPython已经足够成熟,可以放心使用。关于更多IronPython的信息,可以参考在QCon London 20092中,《IronPython in Action》一书的作者Michael Foord的演讲“Real World IronPython”。

在这里,我们先准备一个简单的aspx页面,其中有两个文本框,以及一个按钮:

<asp:TextBox ID="txtCode" runat="server" Height="320px" TextMode="MultiLine" 
    Width="640px"></asp:TextBox>
<br />
<asp:Button ID="btnExecute" runat="server" onclick="btnExecute_Click" 
    Text="Execute" />
<br />
<asp:TextBox ID="txtOutput" runat="server" Height="320px" TextMode="MultiLine"
    Width="640px"></asp:TextBox>

这里,我们希望在点击按钮之后,可以执行txtCode中的IronPython代码,并且将信息显示在txtOutput中:

protected void btnExecute_Click(object sender, EventArgs e)
{
    ScriptEngine engine = Python.CreateEngine();
    var scope = engine.CreateScope();
    var script = engine.CreateScriptSourceFromString(
        this.txtCode.Text, SourceCodeKind.Statements);
    script.Execute(scope);

    TextWriter writer = new StringWriter();
    scope.SetVariable("logger", writer);

    Action<HttpContext> trace;
    if (scope.TryGetVariable<Action<HttpContext>>("trace", out trace))
    {
        trace(this.Context);
    }

    this.txtOutput.Text = writer.ToString();
}

如果您要使用上面的代码,则需要引用IronPython、Microsoft.Scripting及Microsoft.Scripting.Core三个程序集。您可能没有接触过这部分内容,但是应该也能够轻易理解上述代码的含义:

  1. 创建一个ScriptEngine对象,并得到一个ScriptScope。
  2. 将txtCode文本框内的代码,作为Statements编译为ScriptSource对象。
  3. 创建一个StringWriter对象,并赋给ScriptScope中的logger变量。
  4. 执行ScriptSource,并从中获取trace方法成为一个委托。
  5. 将当前HttpContext传入trace方法执行,并输出内容。

于是我们准备一段IronPython代码,它的作用是输出当前请求的UserAgent,如下图:

只是这么一个简单的功能,我们便可以做很多事情,例如使用反射改变某个对象的状态,甚至调用系统中某个方法进行复杂的工作。不过在实际使用时我们可能还需要额外的注意:由于我们基于单个“请求”,因此只能有一台机器得以动态执行IronPython代码。如果在产品环境中只有一台机器,那么自当无事,否则我们就可能需要某种手段来确保我们的代码是在哪台服务器上执行的。例如,对于DNS轮询方式的附载均衡策略,您只需修改本机DNS即可,否则您可能就需要根据您的NLB策略进行其它一些配置。更进一步,如果您需要代码在所有机器上执行,可能就需要编写一个远程调用平台,可以将代码发送到所有机器上执行,并将结果进行聚合。

在下一篇文章中,我们将使用类似的做法,对系统中的请求进行采样,以获取更加丰富的数据——那是一件更有意思的事情.

 

注1:Crack.NET是用于分析WinForm和WPF应用程序状态的工具,可能是由于CLR托管方式的不同,它无法用于ASP.NET或SQL Server等托管程序上。

注2:(广告时间)QCon北京是中国举办的第一次QCon大会,业界各顶尖专家齐聚一堂,如果哪位朋友或所在公司感兴趣,请和老赵联系。

 

相关文章:使用IronPython检测ASP.NET程序状况(下)

Creative Commons License

本文基于署名 2.5 中国大陆许可协议发布,欢迎转载,演绎或用于商业目的,但是必须保留本文的署名赵劼(包含链接),具体操作方式可参考此处。如您有任何疑问或者授权方面的协商,请给我留言

Add your comment

28 条回复

  1. xuefly
    *.*.*.*
    链接

    xuefly 2009-03-23 00:12:00

    抢个沙发
    睡觉!

  2. 老赵
    admin
    链接

    老赵 2009-03-23 00:15:00

    --引用--------------------------------------------------
    xuefly: 抢个沙发
    睡觉!
    --------------------------------------------------------
    好吧,其实下一篇文章更有意思一些。

  3. 青羽
    *.*.*.*
    链接

    青羽 2009-03-23 00:19:00

    哈哈 刚才还和人讨论小蟒蛇。
    老赵的速度真是惊人啊,刚讨论完那事,又发技术文章了。

  4. 老赵
    admin
    链接

    老赵 2009-03-23 00:30:00

    --引用--------------------------------------------------
    青羽: 哈哈 刚才还和人讨论小蟒蛇。
    老赵的速度真是惊人啊,刚讨论完那事,又发技术文章了。
    --------------------------------------------------------
    其实接触东西多了以后,你会发现想法也会越来越多,所以可写的东西也是越来越多了。

  5. 木野狐(Neil Chen)
    *.*.*.*
    链接

    木野狐(Neil Chen) 2009-03-23 08:03:00

    老赵也搞“非主流”了,呵呵,这个我喜欢。

  6. 青羽
    *.*.*.*
    链接

    青羽 2009-03-23 08:54:00

    @Jeffrey Zhao
    记得在上次的Ajax群英会上,周爱民说一年学一种新语言。
    多接触些东西肯定是有好处的。

  7. 老赵
    admin
    链接

    老赵 2009-03-23 09:06:00

    --引用--------------------------------------------------
    木野狐(Neil Chen): 老赵也搞“非主流”了,呵呵,这个我喜欢。
    --------------------------------------------------------
    老赵从来搞非主流了,只是没有啥新想法,有新想法就会写的……

  8. 老赵
    admin
    链接

    老赵 2009-03-23 09:07:00

    --引用--------------------------------------------------
    青羽: @Jeffrey Zhao
    记得在上次的Ajax群英会上,周爱民说一年学一种新语言。
    多接触些东西肯定是有好处的。
    --------------------------------------------------------
    当然也有不同说法,说一年一种新语言是在浪费时间。

  9. CowNew开源团队
    *.*.*.*
    链接

    CowNew开源团队 2009-03-23 09:16:00

    在我心中只有Python才是真正的Python,IronPython、Jython都是Python-Like的.net或者Java。

  10. 老赵
    admin
    链接

    老赵 2009-03-23 09:21:00

    @CowNew开源团队
    呵呵,它们都符合Python的spec为什么就不是python了?
    语言说白了就是对字符串的处理,让计算机可以运行,为什么用C来作这件事情就叫“正统”,用C#或Java来实现就叫“偏门”了呢?

  11. CowNew开源团队
    *.*.*.*
    链接

    CowNew开源团队 2009-03-23 09:30:00

    --引用--------------------------------------------------
    Jeffrey Zhao: @CowNew开源团队
    呵呵,它们都符合Python的spec为什么就不是python了?
    语言说白了就是对字符串的处理,让计算机可以运行,为什么用C来作这件事情就叫“正统”,用C#或Java来实现就叫“偏门”了呢?
    --------------------------------------------------------
    Python的那些模块Jython、IronPython能调用吗?不能调用的话就没意义。

  12. 老赵
    admin
    链接

    老赵 2009-03-23 09:42:00

    @CowNew开源团队
    模块是指什么,Python自带类库?py文件?当然都可以执行。
    这些都不是玩具,都是业界搞出来的有价值的东西,放心使用吧。

  13. CowNew开源团队
    *.*.*.*
    链接

    CowNew开源团队 2009-03-23 09:51:00

    --引用--------------------------------------------------
    Jeffrey Zhao: @CowNew开源团队

    模块是指什么,Python自带类库?py文件?当然都可以执行。

    这些都不是玩具,都是业界搞出来的有价值的东西,放心使用吧。
    --------------------------------------------------------
    模块(module),Python内置的基础模块肯定这些Python-Like的库都提供兼容支持了,我指的是第三方扩展的module,比如Tk、Cryptographic、msilib、curses等。还有我以前用Python写的使用Python-DB API访问数据库的python能不用改成ADO.net、JDBC就能直接运行吗?

  14. 老赵
    admin
    链接

    老赵 2009-03-23 09:58:00

    @CowNew开源团队
    C编写的这是不行的。
    不过这并不表明IronPython不是个Python实现了啊,就像Java是语言,但是不兼容Java类库的C#也是语言,呵呵。
    IronPython还可以和.NET类库无缝继承,这个也是个宝库。

  15. CowNew开源团队
    *.*.*.*
    链接

    CowNew开源团队 2009-03-23 10:04:00

    --引用--------------------------------------------------
    Jeffrey Zhao: @CowNew开源团队
    这是不行的。
    不过这并不表明IronPython不是个Python实现了啊,就像Java是语言,但是不兼容Java类库的C#也是语言,呵呵。
    IronPython还可以和.NET类库无缝继承,这个也是个宝库。
    --------------------------------------------------------
    呵呵,所以只能说IronPython、Jython之于Python就像当年J++之于Java或者Mono之于.net了。
    早上好,嘿嘿!

  16. 郑晖
    *.*.*.*
    链接

    郑晖 2009-03-23 10:06:00

    能问博主一个问题吗?你的博文经常变更发表时间,相当于一篇文章占两次首页。本人初来乍到,不知其合理性何在?请指教。

  17. 老赵
    admin
    链接

    老赵 2009-03-23 10:09:00

    @CowNew开源团队
    你非要这么说那也没有办法。
    不过其实是不一样的,因为J++和C#是同类产品,打不过的,所以很少有人会用J++,而是用C#,不用C#就用直接用Java了。
    而IronPython不一样,它和C#不是一个东西,利用Python语言的特性,又能直接调用.NET类库,多好。

  18. 老赵
    admin
    链接

    老赵 2009-03-23 10:10:00

    --引用--------------------------------------------------
    郑晖: 能问博主一个问题吗?你的博文经常变更发表时间,相当于一篇文章占两次首页。本人初来乍到,不知其合理性何在?请指教。
    --------------------------------------------------------
    没啥合理不合理的,只是用Windows Live Writer修改文章,保存草稿再发布,时间就更新了。不过也好,顶到前面去看的人多一些,哈哈。

  19. Indigo Dai
    *.*.*.*
    链接

    Indigo Dai 2009-03-23 12:08:00

    老赵也关注动态语言啊!

  20. 老赵
    admin
    链接

    老赵 2009-03-23 12:12:00

    --引用--------------------------------------------------
    Indigo Dai: 老赵也关注动态语言啊!
    --------------------------------------------------------
    老赵关注无数东西……

  21. brightwang
    *.*.*.*
    链接

    brightwang 2009-03-23 15:00:00

    问老赵下,如果现在在项目开发中使用IronPython时机成熟吗?

  22. 老赵
    admin
    链接

    老赵 2009-03-23 15:05:00

    --引用--------------------------------------------------
    brightwang: 问老赵下,如果现在在项目开发中使用IronPython时机成熟吗?
    --------------------------------------------------------
    IronPython已经足够成熟了,也已经有很多案例,不过关键还是看你怎么用它,你用的场景适不适合IronPython。

  23. 佩恩
    *.*.*.*
    链接

    佩恩 2009-03-23 23:56:00

    看了老赵的文章,我测试了在OA中使用IronPython做二次开发的脚本,发现技术上可行波,而且比C#的动态编译、Lua.net都要好用

  24. 老赵
    admin
    链接

    老赵 2009-03-24 00:05:00

    --引用--------------------------------------------------
    佩恩: 看了老赵的文章,我测试了在OA中使用IronPython做二次开发的脚本,发现技术上可行波,而且比C#的动态编译、Lua.net都要好用
    --------------------------------------------------------
    能否介绍一下Lua.NET?

  25. 佩恩
    *.*.*.*
    链接

    佩恩 2009-03-24 10:23:00

    乱说一通:
    Lua.net是开源社区对Lua做的一个Net封装,使用起来要注册很多接口(函数)才能在Lua脚本操纵系统内部功能。当然Lua的目标是做C、C++的脚本,可能用于Wow更合适

    但IronPython则由于与Net系统有很好的集成性(使用相同的程序域?),开发起来就方便很多了,如:Asp.net中只需要传入一个Page对象给IronPython,Py脚本里就可通过Page操纵内部的所有控件及Session等对象。有点象Excel的VBA里,给你一个App,你就可以操纵一切了。

  26. 老赵
    admin
    链接

    老赵 2009-03-24 10:40:00

    @佩恩
    嗯,昨天网上查了一下也只发现了这个,原本以为也有像IronPython一样基于.NET实现的Lua呢。

  27. xuzh[未注册用户]
    *.*.*.*
    链接

    xuzh[未注册用户] 2009-03-31 15:25:00

    Martin Fowler 面向对象的信徒

  28. image72
    218.81.250.*
    链接

    image72 2010-06-29 13:10:22

    为什么visual studio提示无法编译 ?

    如何解决 ?

发表回复

登录 / 登录并记住我 ,登陆后便可删除或修改已发表的评论 (请注意保留评论内容)

昵称:(必填)

邮箱:(必填,仅用于Gavatar

主页:(可选)

评论内容(大于5个字符):

  1. Your Name yyyy-MM-dd HH:mm:ss

使用Live Messenger联系我