好了,前边我们把核心内容介绍完了,接下来要做的就是拦截用户的请求,并把请求转向沙箱内。

  这里我们准备通过实现一个HttpModule类来完成请求的拦截与转发。新建一个HuberHttpModule类,并继承IHttpModule。下面我们暂时只处理Application_BeginRequest事件。

  先获取request和response对象

    HttpApplication application = sender as HttpApplication;
HttpResponse respond = application.Response;
HttpRequest request = application.Request;

  接下来获取当前请求的url:

string url = request.Url.AbsolutePath.ToString();

  我们做了一个约定,所有模块都放在plugins目录中,即需要判断一下当前url是否以“/plugins/”开头,再判断一下是否为静态文件(通常情况下,我们的action命名是不允许包含“.”的)。

  接下来定义一个UrlPathEntity类,把url转换成UrlPathEntity的实例对象,用来存放url对应的插件名称、插件版本、controller、action。

  

    public class UrlPathEntity
{
/// <summary>
/// //插件名称
/// </summary>
public string pluginname { get; set; }
/// <summary>
/// //插件版本
/// </summary>
public int pluginversion { get; set; }
/// <summary>
/// //控制器名称(包含area)
/// </summary>
public string controller { get; set; }
/// <summary>
/// //action名称
/// </summary>
public string action { get; set; }
}

  下面是url转换方法:  

/// <summary>url解析成对象
/// </summary>
/// <param name="url"></param>
/// <param name="isPlugin">是否为插件</param>
/// <returns></returns>
public static UrlPathEntity getUrlPathEntity(string url, bool isPlugin)
{
UrlPathEntity result = null;
var matchs = url.Split(new char[] { '/' }, StringSplitOptions.RemoveEmptyEntries);
if (isPlugin)
{
//var matchs = PluginRgx.Matches(url);
if (matchs != null && matchs.Length > 0)
{
int _index = 0;
result = new UrlPathEntity();
result.pluginname = matchs[_index++];//插件名称
string _pluginversion = matchs[_index++];//插件版本
int pluginversion = -1;
int.TryParse(_pluginversion, out pluginversion);
result.pluginversion = pluginversion;
string urltemp = "/" + result.pluginname;
for (; _index < matchs.Length - 1;)
{
urltemp += "/" + matchs[_index++];
}
result.action = matchs[_index];//action名称
urltemp += "/" + result.action; CAModel controller = null;//控制器名称(包含area)
UrlRefAction.TryGetValue(urltemp.ToLower(), out controller);
if (controller != null)
{
result.controller = controller.ControllerName.Replace("/", ".");
result.action = controller.ActionName;
}
}
}
else
{
if (matchs != null && matchs.Length > 0)
{
int _index = 0;
result = new UrlPathEntity();
result.controller = string.Empty;//控制器名称(包含area)
for (; _index < matchs.Length - 1;)
{
result.controller += "." + matchs[_index++];
}
result.controller = result.controller.Substring(1);
result.action = matchs[_index];//action名称
}
}
return result;
}

  获取转换后的对象:

var urlEntity = HuberPluginHandle.getUrlPathEntity(url.Substring(8), true);

  根据该对象找到对应的沙箱:

 SandBoxDynamicLoader sandBox = HuberPluginHandle.getSandBox(urlEntity.pluginname, urlEntity.pluginversion);

  我们再把请求中携带的参数打包:

RefRequestEntity paras = new RefRequestEntity();
RequestHandle.FillCorRefEntity(paras, request);

  好了,准备工作做好了,最后一步,沙箱调用:  

 var result = sandBox.InvokeMothod(urlEntity.controller, urlEntity.action, paras);

  这个result就是我们想要的结果了,接下来我们要做的就是把这个result返回了

RequestHandle.ResposeResult(respond, result);

  至此,我们自定义的请求管道就算完成了,这其中为了防止通篇代码带来的反感,就省略了好多辅助业务,希望大家谅解。

本来还想把权限管理什么的些进来,后来想了想,这个系列主要是讲框架的原理,而且权限那块是基于sqlite写的,不具有规模并发能力,代码已经开源出来了,地址:https://github.com/Eric-zsp/Huber.net

这个系列暂时先写到这里吧。

转载请注明出处:http://www.cnblogs.com/eric-z/p/5108862.html 

第五篇 基于.net搭建热插拔式web框架(拦截器---请求管道)的更多相关文章

  1. 第三篇 基于.net搭建热插拔式web框架(重造Controller)

    由于.net MVC 的controller 依赖于HttpContext,而我们在上一篇中的沙箱模式已经把一次http请求转换为反射调用,并且http上下文不支持跨域,所以我们要重造一个contro ...

  2. 第二篇 基于.net搭建热插拔式web框架(沙箱的构建)

    上周五写了一个实现原理篇,在评论中看到有朋友也遇到了我的问题,真的是有种他乡遇知己的感觉,整个系列我一定会坚持写完,并在最后把代码开源到git中.上一篇文章很多人看了以后,都表示不解,觉得不知道我到底 ...

  3. 第四篇 基于.net搭建热插拔式web框架(RazorEngine实现)

    在开头也是先给大家道个歉,由于最近准备婚事导致这篇文章耽误了许久,同时也谢谢老婆大人对我的支持. 回顾上篇文章,我们重造了一个controller,这个controller中用到了视图引擎,我们的视图 ...

  4. 基于.net搭建热插拔式web框架(实现原理)

    第一节:我们为什么需要一个热插拔式的web框架? 模块之间独立开发 假设我们要做一个后台管理系统,其中包括“用户活跃度”.“产品管理”."账单管理"等模块.每个模块中有自己的业务特 ...

  5. net搭建热插拔式web框架

    net搭建热插拔式web框架(重造Controller) 由于.net MVC 的controller 依赖于HttpContext,而我们在上一篇中的沙箱模式已经把一次http请求转换为反射调用,并 ...

  6. net搭建热插拔式web框架(沙箱的构建)

    net搭建热插拔式web框架(沙箱的构建) 上周五写了一个实现原理篇,在评论中看到有朋友也遇到了我的问题,真的是有种他乡遇知己的感觉,整个系列我一定会坚持写完,并在最后把代码开源到git中.上一篇文章 ...

  7. 带你手写基于 Spring 的可插拔式 RPC 框架(一)介绍

    概述 首先这篇文章是要带大家来实现一个框架,听到框架大家可能会觉得非常高大上,其实这和我们平时写业务员代码没什么区别,但是框架是要给别人使用的,所以我们要换位思考,怎么才能让别人用着舒服,怎么样才能让 ...

  8. 转-基于NodeJS的14款Web框架

    基于NodeJS的14款Web框架 2014-10-16 23:28 作者: NodeJSNet 来源: 本站 浏览: 1,399 次阅读 我要评论暂无评论 字号: 大 中 小 摘要: 在几年的时间里 ...

  9. 两个基于C++/Qt的开源WEB框架

    1.tufao 项目地址: https://github.com/vinipsmaker/tufao 主页: http://vinipsmaker.github.io/tufao/ 介绍: Tufão ...

随机推荐

  1. jpa+springmvc+springdata(一)

    学习尚硅谷笔记: 首先配置application.xml: <?xml version="1.0" encoding="UTF-8"?> <b ...

  2. Log4j配置详解(转)

    一.Log4j简介 Log4j有三个主要的组件:Loggers(记录器),Appenders (输出源)和Layouts(布局).这里可简单理解为日志类别,日志要输出的地方和日志以何种形式输出.综合使 ...

  3. webstorm对WebGL自动提示

    默认竟然没有勾选上,怪不得提示的时候,有很多webgl接口找不到方法(虽然可以运行).

  4. [LeetCode] Generalized Abbreviation 通用简写

    Write a function to generate the generalized abbreviations of a word. Example: Given word = "wo ...

  5. [LeetCode] Contains Duplicate 包含重复值

    Given an array of integers, find if the array contains any duplicates. Your function should return t ...

  6. [LeetCode] Binary Tree Level Order Traversal II 二叉树层序遍历之二

    Given a binary tree, return the bottom-up level order traversal of its nodes' values. (ie, from left ...

  7. NPOI操作EXCEL(四)——反射机制批量导出excel文件

    前面我们已经实现了反射机制进行excel表格数据的解析,既然有上传就得有下载,我们再来写一个通用的导出方法,利用反射机制实现对系统所有数据列表的筛选结果导出excel功能. 我们来构想一下这样一个画面 ...

  8. oracle日常——sqlplus客户端登录

    1.进入cmd 2.命令--sqlplus--提示输入帐号与密码 3.进入后,就可以直接键入sql命令 ps.sql命令后面需要添加分号后才可以回车执行

  9. 如何解决ajax重复提交的问题

    如下一段代码: 先忽略我没引jquery.js的问题,这是一个案例. 当我们点击提交时,控制台输出两次e,在network里查看,可以看到我们的ajax传输了两次,造成了数据重复提交. 一种解释为bu ...

  10. Android动态加载框架汇总

    几种动态加载的比较 1.Tinker 用途:热修复 GitHub地址:https://github.com/Tencent/tinker/ 使用:http://www.jianshu.com/p/f6 ...