gem 'rack' require 'rack/utils' gem 'tidy' require 'tidy' module Rack class HTMLTidy include Rack::Utils FORMAT = %{|| Tidy: %s - [%s] "%s %s%s %s"\n%s} def initialize(app, opts={}) @app = app @errors = opts[:errors] || false @diagnostics = opts[:diagnostics] || false @logger = opts[:logger] @path = opts[:path] || "/usr/lib/libtidy-0.99.so.0" ::Tidy.path = @path end def call(env) status, headers, response = @app.call(env) headers = HeaderHash.new(headers) if !STATUS_WITH_NO_ENTITY_BODY.include?(status) && !headers['transfer-encoding'] && headers['content-type'] && headers['content-type'].include?("text/html") ::Tidy.open(:show_warnings => true, "char-encoding" => "utf8") do |tidy| html = "" response.each { |part| html += part } tidy.clean(html) log(env, tidy, "errors") if @errors && tidy.errors.length > 0 log(env, tidy, "diagnostics") if @diagnostics && tidy.diagnostics.length > 0 end end [status, headers, response] end private def log(env, tidy, what) now = Time.now logger = @logger || env['rack.errors'] logger.write FORMAT % [ what, now.strftime("%d/%b/%Y %H:%M:%S"), env["REQUEST_METHOD"], env["PATH_INFO"], env["QUERY_STRING"].empty? ? "" : "?"+env["QUERY_STRING"], env["HTTP_VERSION"], tidy.send(what) ] end end end