module ActionDispatch::Http::Cache::Request

def etag_matches?(etag)

def etag_matches?(etag)
  if etag
    validators = if_none_match_etags
    validators.include?(etag) || validators.include?("*")
  end
end

def fresh?(response)

Reference: http://tools.ietf.org/html/rfc7232#section-6
`config.action_dispatch.strict_freshness`.
or both are considered equally. You can adjust the preference with
If both headers are supplied, based on configuration, either `ETag` is preferred over `Last-Modified`
`If-Modified-Since` and `If-None-Match` conditions.
Check response freshness (`Last-Modified` and `ETag`) against request
def fresh?(response)
  if Request.strict_freshness
    if if_none_match
      etag_matches?(response.etag)
    elsif if_modified_since
      not_modified?(response.last_modified)
    else
      false
    end
  else
    last_modified = if_modified_since
    etag          = if_none_match
    return false unless last_modified || etag
    success = true
    success &&= not_modified?(response.last_modified) if last_modified
    success &&= etag_matches?(response.etag) if etag
    success
  end
end

def if_modified_since

def if_modified_since
  if since = get_header(HTTP_IF_MODIFIED_SINCE)
    Time.rfc2822(since) rescue nil
  end
end

def if_none_match

def if_none_match
  get_header HTTP_IF_NONE_MATCH
end

def if_none_match_etags

def if_none_match_etags
  if_none_match ? if_none_match.split(",").each(&:strip!) : []
end

def not_modified?(modified_at)

def not_modified?(modified_at)
  if_modified_since && modified_at && if_modified_since >= modified_at
end