aboutsummaryrefslogtreecommitdiff
path: root/src/guff/handlers/guff-stuff.cr
blob: cc3ac6c50616e763202ea01905dc9052095c09cc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
require "uri"
require "../handler"
require "../mime-type"
require "../views/html/page"

class Guff::Handlers::GuffStuffHandler < Guff::Handler
  def initialize(models)
    super(models)
    @digests = {} of String => String
  end

  def call(context : HTTP::Server::Context)
    path = context.request.path.not_nil!

    if matching_request?(context.request.method, path)
      reply(context)
    else
      call_next(context)
    end
  end

  private def reply(context : HTTP::Server::Context)
    # get expanded path to file
    if path = expand_path(context.request.path.not_nil!)
      # get file digest
      file_digest = digest(path)

      # check for cache header
      if context.request.headers["if-none-match"]? == file_digest
        # cached, send 304 not modified
        context.response.status_code = 304
      else
        # not cached, set code and send headers
        context.response.status_code = 200
        context.response.content_type = Guff::MimeType.mime_type(path)
        context.response.content_length = File.size(path)
        context.response.headers["etag"] = file_digest

        if context.request.method == "GET"
          # send body
          File.open(path) do |fh|
            IO.copy(fh, context.response)
          end
        end
      end
    else
      # file not found
      context.response.status_code = 404
    end
  end

  VALID_METHODS = %w{GET HEAD}
  PATH_RE = %r{^/guff-stuff/}

  private def matching_request?(method, path)
    VALID_METHODS.includes?(method) && PATH_RE.match(path)
  end

  private def expand_path(req_path : String) : String?
    # unescape path, check for nil byte
    path = URI.unescape(req_path)
    return nil if path.includes?('\0')

    # build absolute path
    r = File.join(
      @models.config["stuff"],
      File.expand_path(path.gsub(PATH_RE, ""), "/")
    )

    # return path if file exists, or nil otherwise
    File.file?(r) ? r : nil
  end

  private def digest(path : String) : String
    @digests[path] ||= OpenSSL::Digest.new("SHA1").file(path).hexdigest
  end
end