在尝试把MVC3项目部署在mono(2.10)上时,发现两个问题,一个是MVC3所有Action中来自ModelBinder的实体都为null,另一个是Lucene.Net遭遇“UNC paths should be of the form \\server\share”错误。
对于问题1,经过实验,发现mono下只要是post请求,就无法通过Request获取值,因为Request.Form.Keys.Count总是为0,抛开apache直接运行xsp4也存在同样的问题,由于MVC版HttpContext的获取在MVC3的相关类库中进行,而System.Web.Mvc是私有部署的,所以问题要么发生在mono的System.Web的实现中,要么发生在XSP中,经过了痛苦的搜索,有人提到mono中不能使用DefaultModelBinder,也有人提到mono当前还不足以完全支持MVC3,但如果HttpContext中根本就没有初始化POST请求的数据,无论谁的ModelBinder都无法拼出一个Model来。后来,终于在mono的BUG更正列表中发现了一些蛛丝马迹,在mono 2.10.2的页,BUG683339提到:
683339: POST variables are not transferred to HttpContext.Request.Params nor FormCollection in MVC3 app
怪不得,mono2.10.2之前的版本中,POST数据都无法在MVC3中读取,在更新了mono2.10.2后,ModelBinder果然正常工作。
对于问题2,则是windows和linux下路径的"\"和"/"的差异引起的IO异常,其实从windows环境下迁移到linux+mono中时,有80%的问题都集中在URL大小写和windows/linux的路径问题上。这个异常向上追踪可在Stack Trace中看到这样一条信息:
at PanGu.Dict.WordDictionary.LoadFromBinFile (string,string&)
推测是盘古分词在读取词典时的路径中使用了硬编码的路径分隔符,通过Reflector定位到该方法,果然看到了这样的代码:
public string GetDictionaryPath() { string dictionaryPath = this.DictionaryPath; string currentDirectory = Directory.GetCurrentDirectory(); Directory.SetCurrentDirectory(Path.GetAssemblyPath()); dictionaryPath = Path.GetFullPath(dictionaryPath); Directory.SetCurrentDirectory(currentDirectory); return Path.AppendDivision(dictionaryPath, '\\'); //<----------此处使用了硬编码的路径分隔符 }
于是下载了盘古分词的源码,更改其中的词典路径为从配置文件中读取,重新编辑部署了Pangu.dll后,果然可以正常搜索。这也教训我们,应该养成良好习惯,尽可能少地使用硬编码,多考虑代码在多环境下的适用性。
以上是使用mono的点滴Debug经验,可能有很多人也会遇到同样的问题,希望有所帮助。