美文网首页Ruby程序员
Ruby进阶之Rack入门

Ruby进阶之Rack入门

作者: 感觉被掏空 | 来源:发表于2016-12-27 21:39 被阅读784次

    简介

    基本上所有的 Ruby web framework 都是Rack App,web框架大多都是基于rack之上的,比如Ruby On Rails、Sinatra.......,Rack帮助我们封装处理了很多基本的操作,并提供了大量工具

    Rack

    安装 gem

    gem install rack
    

    如何使用

    • rack 会将请求参数封装成为一个Hash对象,叫做 env
    • 它需要一个响应 call 方法的对象, 接受 env 参数
    • 处理结束后,需要返回三元素的数组: 分别是 status code, header, body,其中 status code 是HTTP状态码, header 是一个 hash, body 是一个响应 each 方法的对象

    简单尝试

    #! /usr/bin/env ruby
    #  encoding: utf-8
    
    require 'rack'
    
    app = proc { |env| [200, {}, ['hello world']] }
    
    ::Rack::Handler::WEBrick.run app
    
    • 运行上面代码,WEBrick服务器就会启动,访问localhost:8080就能得到一个hello world返回
    • 注意: ruby的块(proc, lambda)都是响应call方法的对象
    • 200是要返回的状态码,响应头我们保持默认,不做设置,故传递 {},响应each方法的对象我们一般传递一个数组,当然,也可以是其他能响应each方法的对象
    分析
    • WEBrick是我们最常见的web容器了吧,但是在这里,并不像我们单独使用WEBrick那样启动,Rack已经将WEBrick托管了,这里由Rack来启动服务器
    • Rack托管的web容器还有很多,比如Thin, Puma, CGI .......
    • 我们可以将Rack定义为一个介于web app 和 web 容器之间的中间件或者接口

    env对象

    #! /usr/bin/env ruby
    #  encoding: utf-8
    
    require 'rack'
    
    class MyApp
      def call(env)
        puts env.inspect
        [200, {'Content-Type' => 'text/plain'}, ['hello world']]
      end
    end
    
    ::Rack::Handler::WEBrick.run MyApp.new
    
    • 如果你运行了以上代码,你可以在console看到env对象的输出
    • env 对象将请求的所有信息进行了封装成了一个hash对象,直接操作hash对象,简化了我们使用的复杂度
    • Rack还提供了很多强大的辅助模块,帮助我们进一步解析env对象,比如Rack::RequestRack::Response模块(后面再讲)

    Rack中间件介绍

    • 由于web应用的复杂性,我们为了代码的封装,不可能在一个对象的call方法中就完成所有操作,我们可能需要将通用的逻辑分离
    • Rack的中间件特性可以让我们链式调用处理对象,直到处理完成,并且方便对中间件的自由组合
    • 另外:Rack中间件的调用方式十分优雅,之后会分析Rack调用原理

    Rack中间件简单使用

    新建一个文件 config.ru

    #! /usr/bin/env ruby
    #  encoding: utf-8
    
    require 'rack'
    
    class MyMiddleware
      def initialize(app)
        @app = app
      end
    
      def call(env)
        code, headers, body = @app.call(env)
        body << ['this is my middleware']
        [code, headers, body]
      end
    end
    
    class MyApp
      def call(env)
        [200, {'Content-Type' => 'text/plain'}, ['hello world']]
      end
    end
    
    use MyMiddleware
    run MyApp.new
    
    • 这次我们在console通过rackup命令启动, rackup是Rack提供的命令,它会默认执行当前目录下config.ru文件,可以不用写require 'rack',并且文件中我们可以直接写run MyApp.new,不用显示写出web容器,Rack会根据当前情况自己选择web容器启动
    • 运行上面的代码,最后得到看到的返回内容,除了hello world, 还有this is my middleware
    • 因为Rack需要一个响应call方法的对象,只要让中间件也响应同样的方法,再配合设计模式中的组合模式,就实现了所谓的Rack中间件
    • Rack其实是将MyApp的对象作为参数传递给了MyMiddleware, 当Rack去调用MyMiddleware的call方法时,也会触发MyApp的call方法
    • 这里建议大家有空看看组合模式,以上相当于在调用MyMiddleware.new(MyApp.new()).call(env)

    Rack提供的路由

    • Rack也提供了路由功能,大概所有框架都是自带一套路由解析规则,不过Rack提供了最基础的路由定义
    • 我们来试试一个简单的路由,当然咯,会比之前的代码复杂一点
    #! /usr/bin/env ruby
    #  encoding: utf-8
    
    require 'rack'
    
    class MyMiddleware
      def initialize(app)
        @app = app
      end
    
      def call(env)
        code, headers, body = @app.call(env)
        body << 'this is my middleware'
        [code, headers, body]
      end
    end
    
    class MyApp
      def call(env)
        [200, {'Content-Type' => 'text/plain'}, ['hello world']]
      end
    end
    
    app = Rack::Builder.app do 
      map '/' do
        run MyApp.new
      end
      map '/middleware' do
        use MyMiddleware
        run MyApp.new
      end
    end
    
    run app
    
    • rackup运行以上代码,两个路由的返回结果肯定是不一样的,一个采用了中间件,一个没有
    • 比起之前的代码,我们创建了一个Rack::Builder对象,里面定义了路由规则,之前没有使用路由时的代码,只是没有显示创建Rack::Builder对象,其实每个Rack App,都是Rack::Builder实例化的对象
    • run 和 use 都是可以多次使用的,方便根据功能灵活选择
    • 注意:在唯一路由中,只应该出现一个 run ,这是我们逻辑处理的入口

    总结

    • Rack封装了web的基本操作,这些操作很基础,但是很繁琐
    • 封装过的env对象,让我们处理http请求信息十分容易
    • 中间件的使用,可以让我们方便地实现代码的热插拔过滤器等等功能
    • 简单的路由也方便了更多的扩展功能

    后面我们会深入Rack更多知识

    相关文章

      网友评论

        本文标题:Ruby进阶之Rack入门

        本文链接:https://www.haomeiwen.com/subject/rouevttx.html