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. 在Linux(Centos7)系统上对进行Hadoop分布式配置以及运行Hadoop伪分布式实例

    在Linux(Centos7)系统上对进行Hadoop分布式配置以及运行Hadoop伪分布式实例                                                     ...

  2. Linux-网络综合实验

    题目要求:财务部门有俩台机器acc-1和acc-2,it部门有俩台机器it-2和it-2,要求acc机器互通,it机器互通,财务部和it部机器通过路由器互通

  3. 精读《Function VS Class 组件》

    1. 引言 为什么要了解 Function 写法的组件呢?因为它正在变得越来越重要. 那么 React 中 Function Component 与 Class Component 有何不同? how ...

  4. 福州大学软件工程1916|W班 第3次作业成绩排名

    作业链接 结对第二次-文献摘要热词统计及进阶需求 评分细则 本次作业由三部分组成(程序满分80,博客满分70,工程能力满分30) 程序评分标准 基础需求 共有7个测试用例,每个满分20分并按照一定的映 ...

  5. Linux虚拟机搭建本地yum源

    Yum本地源的配置 本教程是在虚拟机里安装Red Hat Enterprise Linux 7 ,以其为例使用iso文件进行Yum本地源的配置.所使用的软件如下: (1)虚拟机:Vmware work ...

  6. Android常规布局方式和方法

    一.关于给控件添加ID属性 <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xm ...

  7. Java学习之软件安装

    成功安装了jdk-10.0.1.eclipse-committers-2018-09-win32-x86_64.mysql-5.7.18.1和tomcat-9.0.0.M17

  8. [CF1140C]Playlist

    Description: 给你n首歌,每首歌有一个长度\(a_i\)和美丽度\(b_i\) 现在可以选出最多k首,动听值为\(\sum a_i*min_{\sum b_i}\) Hint: \(n \ ...

  9. 下载Spring4.3.18.RELEASE的官方文档

    wget -p --page-requisites --convert-links -P /root/spring https://docs.spring.io/spring/docs/4.3.18. ...

  10. vue开发(开发环境+项目搭建)

    Vue.js是一套构建用户界面的渐进式框架.与其他重量级框架不同的是,Vue 采用自底向上增量开发的设计.Vue 的核心库只关注视图层,并且非常容易学习,非常容易与其它库或已有项目整合.另一方面,Vu ...