devdocs/lib/app.rb

314 lines
9.7 KiB
Ruby
Raw Normal View History

2016-01-10 17:55:30 +01:00
# frozen_string_literal: true
2013-10-24 20:25:52 +02:00
require 'bundler/setup'
Bundler.require :app
class App < Sinatra::Application
Bundler.require environment
require 'sinatra/cookies'
2015-10-18 17:26:00 +02:00
require 'tilt/erubis'
2013-10-24 20:25:52 +02:00
2014-01-16 04:47:33 +01:00
Rack::Mime::MIME_TYPES['.webapp'] = 'application/x-web-app-manifest+json'
2013-10-24 20:25:52 +02:00
configure do
set :sentry_dsn, ENV['SENTRY_DSN']
set :protection, except: [:frame_options, :xss_header]
set :root, Pathname.new(File.expand_path('../..', __FILE__))
set :sprockets, Sprockets::Environment.new(root)
set :assets_prefix, 'assets'
set :assets_path, -> { File.join(public_folder, assets_prefix) }
set :assets_manifest_path, -> { File.join(assets_path, 'manifest.json') }
set :assets_compile, %w(*.png docs.js docs.json application.js application.css application-dark.css)
2013-10-24 20:25:52 +02:00
require 'yajl/json_gem'
set :docs_prefix, 'docs'
set :docs_host, -> { File.join('', docs_prefix) }
set :docs_path, -> { File.join(public_folder, docs_prefix) }
set :docs_manifest_path, -> { File.join(docs_path, 'docs.json') }
set :docs, -> { Hash[JSON.parse(File.read(docs_manifest_path)).map! { |doc| [doc['slug'], doc] }] }
set :default_docs, %w(css dom dom_events html http javascript)
2013-10-24 20:25:52 +02:00
2014-11-30 19:56:02 +01:00
set :news_path, -> { File.join(root, assets_prefix, 'javascripts', 'news.json') }
set :news, -> { JSON.parse(File.read(news_path)) }
2013-10-24 20:25:52 +02:00
Dir[docs_path, root.join(assets_prefix, '*/')].each do |path|
sprockets.append_path(path)
end
Sprockets::Helpers.configure do |config|
config.environment = sprockets
config.prefix = "/#{assets_prefix}"
config.public_path = public_folder
2015-05-04 04:25:50 +02:00
config.protocol = :relative
2013-10-24 20:25:52 +02:00
end
end
2015-01-03 16:38:22 +01:00
configure :test, :development do
require 'active_support/per_thread_registry'
2013-10-24 20:25:52 +02:00
require 'active_support/cache'
sprockets.cache = ActiveSupport::Cache.lookup_store :file_store, root.join('tmp', 'cache', 'assets')
2015-01-03 16:38:22 +01:00
end
configure :development do
register Sinatra::Reloader
2013-10-24 20:25:52 +02:00
use BetterErrors::Middleware
BetterErrors.application_root = File.expand_path('..', __FILE__)
BetterErrors.editor = :sublime
end
configure :production do
set :static, false
2015-05-04 04:25:50 +02:00
set :docs_host, '//docs.devdocs.io'
2013-10-24 20:25:52 +02:00
use Rack::ConditionalGet
use Rack::ETag
use Rack::Deflater
use Rack::Static,
root: 'public',
urls: %w(/assets /docs/ /images /favicon.ico /robots.txt /opensearch.xml /manifest.webapp),
2013-10-24 20:25:52 +02:00
header_rules: [
2014-12-14 23:42:57 +01:00
[:all, {'Cache-Control' => 'no-cache, max-age=0'}],
2013-10-24 20:25:52 +02:00
['/assets', {'Cache-Control' => 'public, max-age=604800'}],
['/favicon.ico', {'Cache-Control' => 'public, max-age=86400'}],
['/images', {'Cache-Control' => 'public, max-age=86400'}] ]
sprockets.js_compressor = Uglifier.new output: { beautify: true, indent_level: 0 }
sprockets.css_compressor = :sass
Sprockets::Helpers.configure do |config|
config.digest = true
2015-05-04 04:25:19 +02:00
config.asset_host = 'cdn.devdocs.io'
2013-10-24 20:25:52 +02:00
config.manifest = Sprockets::Manifest.new(sprockets, assets_manifest_path)
end
end
configure :test do
set :docs_manifest_path, -> { File.join(root, 'test', 'files', 'docs.json') }
end
2013-10-24 20:25:52 +02:00
helpers do
include Sinatra::Cookies
include Sprockets::Helpers
def browser
@browser ||= Browser.new ua: request.user_agent
end
2015-10-18 17:26:00 +02:00
UNSUPPORTED_IE_VERSIONS = %w(6 7 8 9).freeze
2013-10-24 20:25:52 +02:00
def unsupported_browser?
2015-10-18 17:26:00 +02:00
browser.ie? && UNSUPPORTED_IE_VERSIONS.include?(browser.version)
2013-10-24 20:25:52 +02:00
end
def docs
@docs ||= begin
cookie = cookies[:docs]
2015-10-18 17:26:00 +02:00
if cookie.nil? || cookie.empty?
settings.default_docs
else
cookie.split('/')
end
end
end
def doc_index_urls
2015-10-18 17:26:00 +02:00
docs.each_with_object [] do |slug, result|
2013-10-24 20:25:52 +02:00
if doc = settings.docs[slug]
2014-12-14 23:42:57 +01:00
result << File.join('', settings.docs_prefix, doc['index_path']) + "?#{doc['mtime']}"
2013-10-24 20:25:52 +02:00
end
end
end
def doc_index_page?
@doc && request.path == "/#{@doc['slug']}/"
end
2015-01-03 15:24:07 +01:00
def query_string_for_redirection
request.query_string.empty? ? nil : "?#{request.query_string}"
end
2015-02-08 23:50:03 +01:00
def main_stylesheet_path
2015-08-03 23:06:19 +02:00
stylesheet_paths[dark_theme? ? :dark : :default]
2015-02-08 23:50:03 +01:00
end
def alternate_stylesheet_path
2015-08-03 23:06:19 +02:00
stylesheet_paths[dark_theme? ? :default : :dark]
2015-02-08 23:50:03 +01:00
end
def stylesheet_paths
@stylesheet_paths ||= {
default: stylesheet_path('application'),
dark: stylesheet_path('application-dark')
}
end
2015-08-03 23:06:19 +02:00
def app_size
@app_size ||= cookies[:size].nil? ? '18rem' : "#{cookies[:size]}px"
end
def app_layout
cookies[:layout]
end
2015-08-03 23:06:19 +02:00
def app_theme
@app_theme ||= cookies[:dark].nil? ? 'default' : 'dark'
end
def dark_theme?
app_theme == 'dark'
end
def redirect_via_js(path) # courtesy of HTML5 App Cache
response.set_cookie :initial_path, value: path, expires: Time.now + 15, path: '/'
redirect '/', 302
end
def supports_js_redirection?
browser.modern? && !cookies.empty?
end
2013-10-24 20:25:52 +02:00
end
before do
halt erb :unsupported if unsupported_browser?
end
OUT_HOST = 'out.devdocs.io'.freeze
before do
if request.host == OUT_HOST && !request.path.start_with?('/s/')
query_string = "?#{request.query_string}" unless request.query_string.empty?
redirect "http://devdocs.io#{request.path}#{query_string}", 302
end
end
2013-10-24 20:25:52 +02:00
get '/manifest.appcache' do
content_type 'text/cache-manifest'
2014-12-14 23:42:57 +01:00
expires 0, :'no-cache'
2013-10-24 20:25:52 +02:00
erb :manifest
end
get '/' do
return redirect '/' unless request.query_string.empty? # courtesy of HTML5 App Cache
2013-10-24 20:25:52 +02:00
erb :index
end
%w(offline about news help).each do |page|
2013-10-24 20:25:52 +02:00
get "/#{page}" do
if supports_js_redirection?
redirect_via_js "/#{page}"
else
redirect "/#/#{page}", 302
end
2013-10-24 20:25:52 +02:00
end
end
2014-02-08 17:32:36 +01:00
get '/search' do
redirect "/#q=#{params[:q]}"
end
2013-10-24 20:25:52 +02:00
get '/ping' do
200
end
%w(docs.json application.js application.css).each do |asset|
class_eval <<-CODE, __FILE__, __LINE__ + 1
get '/#{asset}' do
redirect asset_path('#{asset}', protocol: 'http')
end
CODE
end
2015-07-13 04:53:47 +02:00
{
'/s/maxcdn' => 'https://www.maxcdn.com/?utm_source=devdocs&utm_medium=banner&utm_campaign=devdocs',
'/s/shopify' => 'https://www.shopify.com/careers?utm_source=devdocs&utm_medium=banner&utm_campaign=devdocs',
'/s/jetbrains' => 'https://www.jetbrains.com/?utm_source=devdocs&utm_medium=sponsorship&utm_campaign=devdocs',
'/s/jetbrains/ruby' => 'https://www.jetbrains.com/ruby/?utm_source=devdocs&utm_medium=sponsorship&utm_campaign=devdocs',
'/s/jetbrains/python' => 'https://www.jetbrains.com/pycharm/?utm_source=devdocs&utm_medium=sponsorship&utm_campaign=devdocs',
'/s/jetbrains/c' => 'https://www.jetbrains.com/clion/?utm_source=devdocs&utm_medium=sponsorship&utm_campaign=devdocs',
'/s/jetbrains/web' => 'https://www.jetbrains.com/webstorm/?utm_source=devdocs&utm_medium=sponsorship&utm_campaign=devdocs',
'/s/code-school' => 'http://www.codeschool.com/?utm_campaign=devdocs&utm_content=homepage&utm_source=devdocs&utm_medium=sponsorship',
'/s/tw' => 'https://twitter.com/intent/tweet?url=http%3A%2F%2Fdevdocs.io&via=DevDocs&text=All-in-one%2C%20offline%20API%20documentation%20browser%3A',
2016-01-09 16:44:57 +01:00
'/s/fb' => 'https://www.facebook.com/sharer/sharer.php?u=http%3A%2F%2Fdevdocs.io',
2015-07-13 04:53:47 +02:00
'/s/re' => 'http://www.reddit.com/submit?url=http%3A%2F%2Fdevdocs.io&title=All-in-one%2C%20offline%20API%20documentation%20browser&resubmit=true'
}.each do |path, url|
class_eval <<-CODE, __FILE__, __LINE__ + 1
get '#{path}' do
redirect '#{url}'
end
CODE
2014-12-14 18:29:34 +01:00
end
2015-01-03 14:39:25 +01:00
get %r{\A/feed(?:\.atom)?\z} do
2014-11-30 21:55:26 +01:00
content_type 'application/atom+xml'
settings.news_feed
end
2015-01-03 15:15:46 +01:00
get %r{\A/(\w+)(\-[\w\-]+)?(/.*)?\z} do |doc, type, rest|
return 404 unless @doc = settings.docs[doc]
2015-01-03 15:15:46 +01:00
if rest.nil?
2015-01-03 15:24:07 +01:00
redirect "/#{doc}#{type}/#{query_string_for_redirection}"
2015-01-03 15:15:46 +01:00
elsif rest.length > 1 && rest.end_with?('/')
2015-01-03 15:24:07 +01:00
redirect "/#{doc}#{type}#{rest[0...-1]}#{query_string_for_redirection}"
elsif docs.include?(doc) && supports_js_redirection?
redirect_via_js(request.path)
else
erb :other
end
end
2013-10-24 20:25:52 +02:00
not_found do
send_file File.join(settings.public_folder, '404.html'), status: status
end
error do
send_file File.join(settings.public_folder, '500.html'), status: status
end
2014-11-30 21:55:26 +01:00
configure do
require 'rss'
feed = RSS::Maker.make('atom') do |maker|
maker.channel.id = 'tag:devdocs.io,2014:/feed'
maker.channel.title = 'DevDocs'
maker.channel.author = 'DevDocs'
maker.channel.updated = "#{settings.news.first.first}T14:00:00Z"
maker.channel.links.new_link do |link|
link.rel = 'self'
link.href = 'http://devdocs.io/feed.atom'
link.type = 'application/atom+xml'
end
maker.channel.links.new_link do |link|
link.rel = 'alternate'
link.href = 'http://devdocs.io/'
link.type = 'text/html'
end
news.each_with_index do |news, i|
maker.items.new_item do |item|
item.id = "tag:devdocs.io,2014:News/#{settings.news.length - i}"
item.title = news[1].split("\n").first.gsub(/<\/?[^>]*>/, '')
item.description do |desc|
desc.content = news[1..-1].join.gsub("\n", '<br>').gsub('href="/', 'href="http://devdocs.io/')
desc.type = 'html'
end
item.updated = "#{news.first}T14:00:00Z"
item.published = "#{news.first}T14:00:00Z"
item.links.new_link do |link|
link.rel = 'alternate'
link.href = 'http://devdocs.io/'
link.type = 'text/html'
end
end
end
end
set :news_feed, feed.to_s
end
2013-10-24 20:25:52 +02:00
end