app.rb 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. require 'bundler/setup'
  2. Bundler.require :app
  3. class App < Sinatra::Application
  4. Bundler.require environment
  5. require 'sinatra/cookies'
  6. Rack::Mime::MIME_TYPES['.webapp'] = 'application/x-web-app-manifest+json'
  7. configure do
  8. set :sentry_dsn, ENV['SENTRY_DSN']
  9. set :protection, except: [:frame_options, :xss_header]
  10. set :root, Pathname.new(File.expand_path('../..', __FILE__))
  11. set :sprockets, Sprockets::Environment.new(root)
  12. set :assets_prefix, 'assets'
  13. set :assets_path, -> { File.join(public_folder, assets_prefix) }
  14. set :assets_manifest_path, -> { File.join(assets_path, 'manifest.json') }
  15. set :assets_compile, %w(*.png docs.js docs.json application.js application.css application-dark.css)
  16. require 'yajl/json_gem'
  17. set :docs_prefix, 'docs'
  18. set :docs_host, -> { File.join('', docs_prefix) }
  19. set :docs_path, -> { File.join(public_folder, docs_prefix) }
  20. set :docs_manifest_path, -> { File.join(docs_path, 'docs.json') }
  21. set :docs, -> { Hash[JSON.parse(File.read(docs_manifest_path)).map! { |doc| [doc['slug'], doc] }] }
  22. set :default_docs, %w(css dom dom_events html http javascript)
  23. set :news_path, -> { File.join(root, assets_prefix, 'javascripts', 'news.json') }
  24. set :news, -> { JSON.parse(File.read(news_path)) }
  25. Dir[docs_path, root.join(assets_prefix, '*/')].each do |path|
  26. sprockets.append_path(path)
  27. end
  28. Sprockets::Helpers.configure do |config|
  29. config.environment = sprockets
  30. config.prefix = "/#{assets_prefix}"
  31. config.public_path = public_folder
  32. end
  33. end
  34. configure :test, :development do
  35. require 'active_support/per_thread_registry'
  36. require 'active_support/cache'
  37. sprockets.cache = ActiveSupport::Cache.lookup_store :file_store, root.join('tmp', 'cache', 'assets')
  38. end
  39. configure :development do
  40. register Sinatra::Reloader
  41. use BetterErrors::Middleware
  42. BetterErrors.application_root = File.expand_path('..', __FILE__)
  43. BetterErrors.editor = :sublime
  44. end
  45. configure :production do
  46. set :static, false
  47. set :docs_host, 'http://maxcdn-docs.devdocs.io'
  48. use Rack::ConditionalGet
  49. use Rack::ETag
  50. use Rack::Deflater
  51. use Rack::Static,
  52. root: 'public',
  53. urls: %w(/assets /docs/ /images /favicon.ico /robots.txt /opensearch.xml /manifest.webapp),
  54. header_rules: [
  55. [:all, {'Cache-Control' => 'no-cache, max-age=0'}],
  56. ['/assets', {'Cache-Control' => 'public, max-age=604800'}],
  57. ['/favicon.ico', {'Cache-Control' => 'public, max-age=86400'}],
  58. ['/images', {'Cache-Control' => 'public, max-age=86400'}] ]
  59. sprockets.js_compressor = Uglifier.new output: { beautify: true, indent_level: 0 }
  60. sprockets.css_compressor = :sass
  61. Sprockets::Helpers.configure do |config|
  62. config.digest = true
  63. config.asset_host = 'maxcdn.devdocs.io'
  64. config.manifest = Sprockets::Manifest.new(sprockets, assets_manifest_path)
  65. end
  66. end
  67. configure :test do
  68. set :docs_manifest_path, -> { File.join(root, 'test', 'files', 'docs.json') }
  69. end
  70. helpers do
  71. include Sinatra::Cookies
  72. include Sprockets::Helpers
  73. def browser
  74. @browser ||= Browser.new ua: request.user_agent
  75. end
  76. def unsupported_browser?
  77. browser.ie? && %w(6 7 8 9).include?(browser.version)
  78. end
  79. def doc_index_urls
  80. cookie = cookies[:docs]
  81. docs = if cookie.nil? || cookie.empty?
  82. settings.default_docs
  83. else
  84. cookie.split('/')
  85. end
  86. docs.inject [] do |result, slug|
  87. if doc = settings.docs[slug]
  88. result << File.join('', settings.docs_prefix, doc['index_path']) + "?#{doc['mtime']}"
  89. end
  90. result
  91. end
  92. end
  93. def doc_index_page?
  94. @doc && request.path == "/#{@doc['slug']}/"
  95. end
  96. def query_string_for_redirection
  97. request.query_string.empty? ? nil : "?#{request.query_string}"
  98. end
  99. def main_stylesheet_path
  100. stylesheet_paths[cookies[:dark].nil? ? :default : :dark]
  101. end
  102. def alternate_stylesheet_path
  103. stylesheet_paths[cookies[:dark].nil? ? :dark : :default]
  104. end
  105. def stylesheet_paths
  106. @stylesheet_paths ||= {
  107. default: stylesheet_path('application'),
  108. dark: stylesheet_path('application-dark')
  109. }
  110. end
  111. def size
  112. @size ||= cookies[:size].nil? ? '18rem' : "#{cookies[:size]}px"
  113. end
  114. end
  115. before do
  116. halt erb :unsupported if unsupported_browser?
  117. end
  118. get '/manifest.appcache' do
  119. content_type 'text/cache-manifest'
  120. expires 0, :'no-cache'
  121. erb :manifest
  122. end
  123. get '/' do
  124. return redirect '/' unless request.query_string.empty?
  125. erb :index
  126. end
  127. %w(offline about news help).each do |page|
  128. get "/#{page}" do
  129. redirect "/#/#{page}", 302
  130. end
  131. end
  132. get '/search' do
  133. redirect "/#q=#{params[:q]}"
  134. end
  135. get '/ping' do
  136. 200
  137. end
  138. get '/docs.json' do
  139. redirect asset_path('docs.json')
  140. end
  141. get '/s/maxcdn' do
  142. redirect 'https://www.maxcdn.com/?utm_source=devdocs&utm_medium=banner&utm_campaign=devdocs'
  143. end
  144. get '/s/shopify' do
  145. redirect 'http://www.shopify.com/careers?utm_source=devdocs&utm_medium=banner&utm_campaign=devdocs'
  146. end
  147. get %r{\A/feed(?:\.atom)?\z} do
  148. content_type 'application/atom+xml'
  149. settings.news_feed
  150. end
  151. get '/s/tw' do
  152. redirect 'https://twitter.com/intent/tweet?url=http%3A%2F%2Fdevdocs.io&via=DevDocs&text=All-in-one%2C%20offline%20API%20documentation%20browser%3A'
  153. end
  154. get '/s/fb' do
  155. redirect 'https://www.facebook.com/sharer/sharer.php?u=http%3A%2F%2Fdevdocs.io'
  156. end
  157. get '/s/re' do
  158. redirect 'http://www.reddit.com/submit?url=http%3A%2F%2Fdevdocs.io&title=All-in-one%2C%20offline%20API%20documentation%20browser&resubmit=true'
  159. end
  160. get %r{\A/(\w+)(\-[\w\-]+)?(/.*)?\z} do |doc, type, rest|
  161. return 404 unless @doc = settings.docs[doc]
  162. if rest.nil?
  163. redirect "/#{doc}#{type}/#{query_string_for_redirection}"
  164. elsif rest.length > 1 && rest.end_with?('/')
  165. redirect "/#{doc}#{type}#{rest[0...-1]}#{query_string_for_redirection}"
  166. else
  167. erb :other
  168. end
  169. end
  170. not_found do
  171. send_file File.join(settings.public_folder, '404.html'), status: status
  172. end
  173. error do
  174. send_file File.join(settings.public_folder, '500.html'), status: status
  175. end
  176. configure do
  177. require 'rss'
  178. feed = RSS::Maker.make('atom') do |maker|
  179. maker.channel.id = 'tag:devdocs.io,2014:/feed'
  180. maker.channel.title = 'DevDocs'
  181. maker.channel.author = 'DevDocs'
  182. maker.channel.updated = "#{settings.news.first.first}T14:00:00Z"
  183. maker.channel.links.new_link do |link|
  184. link.rel = 'self'
  185. link.href = 'http://devdocs.io/feed.atom'
  186. link.type = 'application/atom+xml'
  187. end
  188. maker.channel.links.new_link do |link|
  189. link.rel = 'alternate'
  190. link.href = 'http://devdocs.io/'
  191. link.type = 'text/html'
  192. end
  193. news.each_with_index do |news, i|
  194. maker.items.new_item do |item|
  195. item.id = "tag:devdocs.io,2014:News/#{settings.news.length - i}"
  196. item.title = news[1].split("\n").first.gsub(/<\/?[^>]*>/, '')
  197. item.description do |desc|
  198. desc.content = news[1..-1].join.gsub("\n", '<br>').gsub('href="/', 'href="http://devdocs.io/')
  199. desc.type = 'html'
  200. end
  201. item.updated = "#{news.first}T14:00:00Z"
  202. item.published = "#{news.first}T14:00:00Z"
  203. item.links.new_link do |link|
  204. link.rel = 'alternate'
  205. link.href = 'http://devdocs.io/'
  206. link.type = 'text/html'
  207. end
  208. end
  209. end
  210. end
  211. set :news_feed, feed.to_s
  212. end
  213. end