Owin中也有类似于ASP.NET的管道,以前在做ASP.NET项目的时候,可以制作很多不同功能HttpHandler或者HttpModule并注册在Web.config中重复使用。在Owin的管道中,我们可以注册中间件(Middleware)来实现相似的功能。

所有的Owin中间件需要继承OwinMiddleware这个抽象类。

public abstract class OwinMiddleware

{

        //构造函数,定义了下一个需要运行的中间件

        protected OwinMiddleware(OwinMiddleware next);

 

        //下一个需要运行的中间件

        protected OwinMiddleware Next { get; set; }

 

        //中间件的执行方法

        public abstract Task Invoke(IOwinContext context);

}

当一个请求发送到Owin服务器之后,Owin服务器会自动获取Request中信息并转换成一个IOwinContext对象,然后依次运行定义在Owin管道中的中间件的Invoke方法。在每个中间件的Invoke方法中,开发人员都可以获取到请求的上下文,并且做出不同的Response。

这里需要注意管道中的中间件是有顺序的,他的顺序就是你在Owin启动类Startup中的注册顺序。每个中间件运行完毕之后,都可以指定是否继续运行后续的中间件。在Invoke方法中可以显示调用Next.Invoke(context)来表明需要继续运行后续的中间件。

下面我们做一个简单的PNG图片水印中间件。

创建项目

首先我们创建一个空的命令行工程PNGWatermark,并依次引入一下Nuget Library

Install-Package Microsoft.Owin.Host    //使用Owin Host托管
Install-Package Microsoft.Owin.StaticFiles    //静态文件中间件

Install-Package Microsoft.Owin.Host.HttpListener //使用HttpListener作为Owin服务器

创建水印中间件

添加一个新类PngWatermarkMiddleware,  其中的AddTextToImg方法是我从网上随便Copy的,这里的主要代码在Invoke方法中。因为Owin中不能使用ASP.NET的Server.MapPath方法,所以这里需要借助PhysicalFileSystem类,来吧url转换成对应的服务器文件路径。在这个方法最后我没有写Next.Invoke(context),  是因为我认为加上水印之后,后续的中间件已经不需要运行了。

public class PngWatermarkMiddleware : OwinMiddleware

    {

        public PngWatermarkMiddleware(OwinMiddleware next) : base(next)

        {

 

        }

 

        public override async Task Invoke(IOwinContext context)

        {

            //获取网站根目录

            PhysicalFileSystem pfs = new PhysicalFileSystem(string.Empty);

            IFileInfo info = null;

 

            //转换url为绝对文件路径, 并检查文件是否存在

            var isFile = pfs.TryGetFileInfo(context.Request.Uri.LocalPath, out info);

 

            if (isFile && info.Name.EndsWith(".png"))

            {

 

                //如果是文件,就读取图片内容,加水印

                Image image = Image.FromFile(info.PhysicalPath);

 

                var imageByteAfterAddingWatermark = AddTextToImg(image, "Author: Lamond");

                context.Response.ContentType = "application/x-png";

                context.Response.StatusCode = (int)HttpStatusCode.OK;

                await context.Response.WriteAsync(imageByteAfterAddingWatermark);

            }

        }

 

        public static byte[] AddTextToImg(Image image, string text)

        {

            Bitmap bitmap = new Bitmap(image, image.Width, image.Height);

            Graphics g = Graphics.FromImage(bitmap);

 

            float fontSize = 12.0f; //字体大小

            float textWidth = text.Length * fontSize; //文本的长度

 

 

            //下面定义一个矩形区域,以后在这个矩形里画上白底黑字

            float rectX = 0;

            float rectY = 0;

            float rectWidth = text.Length * (fontSize + 8);

            float rectHeight = fontSize + 8;

            //声明矩形域

            RectangleF textArea = new RectangleF(rectX, rectY, rectWidth, rectHeight);

 

            //定义字体

            Font font = new Font("宋体", fontSize);

 

            //白笔刷,画文字用

            Brush whiteBrush = new SolidBrush(Color.White);

 

            //黑笔刷,画背景用

            Brush blackBrush = new SolidBrush(Color.Black);

 

            g.FillRectangle(blackBrush, rectX, rectY, rectWidth, rectHeight);

 

            g.DrawString(text, font, whiteBrush, textArea);

 

            MemoryStream ms = new MemoryStream();

            bitmap.Save(ms, ImageFormat.Png);

 

            return ms.ToArray();

        }

    }

注入水印中间件

Owin中间件需要在Owin启动类的Configuration中注册,所以这里我们需要加一个Owin Startup类,代码如下

public class Startup

    {

        public void Configuration(IAppBuilder app)

        {

            app.Use<PngWatermarkMiddleware>();

        }

}

以OwinHost方式启动项目

在Program.cs中加入以下代码,启动Owin服务器

class Program

    {

        static void Main(string[] args)

        {

            var url = "http://localhost:1234";

 

            using (WebApp.Start<Startup>(url))

            {

                Console.WriteLine("Server Started");        

     }

 

     Console.Read();

        }

}

最终运行效果

下面我们将一个logo.png放置到当前项目的bin/debug目录中

然后启动项目,运行效果如下

打开浏览器,输入http://localhost:1234/logo.png

浏览器会自动下载一个logo.png, 打开之后,其内容如下,水印已经成功加上了

Owin学习笔记(二) 中间件开发的更多相关文章

  1. Struts2学习笔记二:开发流程

    一:创建项目,添加依赖包 二:在web.xml配置核心控制器 <filter> <filter-name>struts2</filter-name> <fil ...

  2. Java学习笔记二:Java开发工具Eclipse的安装与使用

    Java开发工具Eclipse的安装与使用 正如office一样我们在开发java语言过程中同样需要依款不错的开发工具,目前市场上的IDE很多,这里只演示Eclipse的安装: 一:下载软件: 1.下 ...

  3. [Firefly引擎][学习笔记二][已完结]卡牌游戏开发模型的设计

    源地址:http://bbs.9miao.com/thread-44603-1-1.html 在此补充一下Socket的验证机制:socket登陆验证.会采用session会话超时的机制做心跳接口验证 ...

  4. amazeui学习笔记二(进阶开发5)--Web 组件开发规范Rules

    amazeui学习笔记二(进阶开发5)--Web 组件开发规范Rules 一.总结 1.见名知意:见那些class名字知意,见函数名知意,见文件名知意 例如(HISTORY.md Web 组件更新历史 ...

  5. amazeui学习笔记二(进阶开发4)--JavaScript规范Rules

    amazeui学习笔记二(进阶开发4)--JavaScript规范Rules 一.总结 1.注释规范总原则: As short as possible(如无必要,勿增注释):尽量提高代码本身的清晰性. ...

  6. amazeui学习笔记二(进阶开发3)--HTML/CSS规范Rules

    amazeui学习笔记二(进阶开发3)--HTML/CSS规范Rules 一.总结 1.am:以 am 为命名空间 2.模块状态: {命名空间}-{模块名}-{状态描述} 3.子模块: {命名空间}- ...

  7. amazeui学习笔记二(进阶开发2)--Web组件简介Web Component

    amazeui学习笔记二(进阶开发2)--Web组件简介Web Component 一.总结 1.amaze ui:amaze ui是一个web 组件, 由模板(hbs).样式(LESS).交互(JS ...

  8. amazeui学习笔记二(进阶开发1)--项目结构structure

    amazeui学习笔记二(进阶开发1)--项目结构structure 一.总结 1.项目结构:是说的amazeui在github上面的项目结构,二次开发amazeui用 二.项目结构structure ...

  9. Qlik Sense学习笔记之Mashup开发(二)

    date: 2019-01-26 11:28:07 updated: 2019-01-26 11:28:07 Qlik Sense学习笔记之Mashup开发(二) 1.Mobile SPA UI Fr ...

  10. MongoDB学习笔记二- Mongoose

    MongoDB学习笔记二 Mongoose Mongoose 简介 之前我们都是通过shell来完成对数据库的各种操作, 在开发中大部分时候我们都需要通过程序来完成对数据库的操作 而Mongoose就 ...

随机推荐

  1. Candies POJ - 3159 (最短路+差分约束)

    During the kindergarten days, flymouse was the monitor of his class. Occasionally the head-teacher b ...

  2. Python-第一篇-python初识及变量

    <work smart>主动分享成果和经验,与同伴共同成长</smart work> 启文:使用20多种编程语言,对大家说节日快乐 万国码使用16位(至少)表示内容: ASCI ...

  3. Java对象转换成Map

    需求总是千奇百怪,对象转成map,看似没必要,但就是有这个需求,所以记录下来 首先是Bean package tools; import lombok.Data; /** * 车辆实体类 */ @Da ...

  4. PCB差分线学习

    问:何为差分信号? 答:通俗地说,就是驱动端发送两个等值.反相的信号,接收端通过比较这两个电压的差值来判断逻辑状态“0”还是“1”. 问:差分线的优势在哪? 答:差分信号和普通的单端信号走线相比,最明 ...

  5. 背包问题——dfs

    背包问题——dfs 问题描述   解决思路 采用DFS搜索 其实也是回溯法 代码实现 #include<iostream> #include<vector> using nam ...

  6. HDU-AcmKeHaoWanLe训练实录

    菜鸡队训练实录. 现场赛记录:[名称:奖项/排名] 2017: ICPC Shenyang:Gold/3 CCPC Hangzhou:Gold/3 ICPC Beijing:Gold/13 CCPC ...

  7. vue-These relative modules were not found

    今天在做vue2.0+webpack的项目的时候,本来一切正常,整理了一下文件夹分类,就是把一些基础的组件新建了一个文件夹移进去,然后就报了以下的错误,其他东西都没改 最后网上找了很多资料,有说配置文 ...

  8. js判断pc还是移动端

    if (!/windows phone|iphone|android/ig.test(window.navigator.userAgent)) { //pc }else{ //h5 }

  9. redis + cookies 实现持久登入

    通过登入把用户信息和token加载到redis中去, 将token和部分用户信息存储在cookie中, 下次登入时 判断cookie的token在redis中是否存在, 存在就把用户信息加载出来自动登 ...

  10. CentOS 7.3/Linux .net core sdk 安装

    执行下列命令,安装.NET Core SDK(微软官方教程地址 https://www.microsoft.com/net/learn/get-started/linuxcentos) 点开链接,选择 ...