-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathrack_static_cache.rb
93 lines (86 loc) · 3.9 KB
/
rack_static_cache.rb
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
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
module Rack
#
# The Rack::StaticCache middleware automatically adds, removes and modifies
# stuffs in response headers to facilitiate client and proxy caching for static files
# that minimizes http requests and improves overall load times for second time visitors.
#
# Once a static content is stored in a client/proxy the only way to enforce the browser
# to fetch the latest content and ignore the cache is to rename the static file.
#
# Alternatively, we can add a version number into the URL to the content to bypass
# the caches. Rack::StaticCache by default handles version numbers in the filename.
# As an example,
# http://yoursite.com/images/test-1.0.0.png and http://yoursite.com/images/test-2.0.0.png
# both reffers to the same image file http://yoursite.com/images/test.png
#
# Another way to bypass the cache is adding the version number in a field-value pair in the
# URL query string. As an example, http://yoursite.com/images/test.png?v=1.0.0
# In that case, set the option :versioning to false to avoid unneccessary regexp calculations.
#
# It's better to keep the current version number in some config file and use it in every static
# content's URL. So each time we modify our static contents, we just have to change the version
# number to enforce the browser to fetch the latest content.
#
# You can use Rack::Deflater along with Rack::StaticCache for further improvements in page loading time.
#
# Examples:
# use Rack::StaticCache, :urls => ["/images", "/css", "/js", "/documents*"], :root => "statics"
# will serve all requests beginning with /images, /csss or /js from the
# directory "statics/images", "statics/css", "statics/js".
# All the files from these directories will have modified headers to enable client/proxy caching,
# except the files from the directory "documents". Append a * (star) at the end of the pattern
# if you want to disable caching for any pattern . In that case, plain static contents will be served with
# default headers.
#
# use Rack::StaticCache, :urls => ["/images"], :duration => 2, :versioning => false
# will serve all requests begining with /images under the current directory (default for the option :root
# is current directory). All the contents served will have cache expiration duration set to 2 years in headers
# (default for :duration is 1 year), and StaticCache will not compute any versioning logics (default for
# :versioning is true)
#
class StaticCache
def initialize(app, options={})
@app = app
@urls = options[:urls]
@no_cache = {}
@urls.collect! do |url|
if url =~ /\*$/
url.sub!(/\*$/, '')
@no_cache[url] = 1
end
url
end
root = options[:root] || Dir.pwd
@file_server = Rack::File.new(root)
@cache_duration = options[:duration] || 1
@versioning_enabled = true
@versioning_enabled = options[:versioning] unless options[:versioning].nil?
@duration_in_seconds = self.duration_in_seconds
@duration_in_words = self.duration_in_words
end
def call(env)
path = env["PATH_INFO"]
url = @urls.detect{ |u| path.index(u) == 0 }
unless url.nil?
path.sub!(/-[\d.]+([.][a-zA-Z][\w]+)?$/, '\1') if @versioning_enabled
status, headers, body = @file_server.call(env)
if @no_cache[url].nil?
headers['Cache-Control'] ="max-age=#{@duration_in_seconds}, public"
headers['Expires'] = @duration_in_words
headers.delete 'Etag'
headers.delete 'Pragma'
headers.delete 'Last-Modified'
end
[status, headers, body]
else
@app.call(env)
end
end
def duration_in_words
(Time.now + self.duration_in_seconds).strftime '%a, %d %b %Y %H:%M:%S GMT'
end
def duration_in_seconds
60 * 60 * 24 * 365 * @cache_duration
end
end
end