class DocsCLI < Thor include Thor::Actions def self.to_s 'Docs' end def initialize(*args) require 'docs' trap('INT') { puts; exit! } # hide backtrace on ^C super end desc 'list', 'List available documentations' def list output = Docs.all.flat_map do |doc| name = doc.to_s.demodulize.underscore if doc.versioned? doc.versions.map { |_doc| "#{name}@#{_doc.version}" } else name end end.join("\n") require 'tty-pager' TTY::Pager.new.page(output) end desc 'page [path] [--version] [--verbose] [--debug]', 'Generate a page (no indexing)' option :version, type: :string option :verbose, type: :boolean option :debug, type: :boolean def page(name, path = '') unless path.empty? || path.start_with?('/') return puts 'ERROR: [path] must be an absolute path.' end Docs.install_report :store if options[:verbose] if options[:debug] GC.disable Docs.install_report :filter, :request end if Docs.generate_page(name, options[:version], path) puts 'Done' else puts "Failed!#{' (try running with --debug for more information)' unless options[:debug]}" end rescue Docs::DocNotFound => error handle_doc_not_found_error(error) end desc 'generate [--version] [--verbose] [--debug] [--force] [--package]', 'Generate a documentation' option :version, type: :string option :verbose, type: :boolean option :debug, type: :boolean option :force, type: :boolean option :package, type: :boolean def generate(name) Docs.install_report :store if options[:verbose] Docs.install_report :scraper if options[:debug] Docs.install_report :progress_bar, :doc if $stdout.tty? unless options[:force] puts <<-TEXT.strip_heredoc Note: this command will scrape the documentation from the source. Some scrapers require a local setup. Others will send thousands of HTTP requests, potentially slowing down the source site. Please don't use it unless you are modifying the code. To download the latest tested version of a documentation, use: thor docs:download #{name}\n TEXT return unless yes? 'Proceed? (y/n)' end if Docs.generate(name, options[:version]) generate_manifest if options[:package] require 'unix_utils' package_doc(Docs.find(name, options[:version])) end puts 'Done' else puts "Failed!#{' (try running with --debug for more information)' unless options[:debug]}" end rescue Docs::DocNotFound => error handle_doc_not_found_error(error) end desc 'manifest', 'Create the manifest' def manifest generate_manifest puts 'Done' end desc 'download ( ... | --default | --installed)', 'Download documentations' option :default, type: :boolean option :installed, type: :boolean option :all, type: :boolean def download(*names) require 'unix_utils' docs = if options[:default] Docs.defaults elsif options[:installed] Docs.installed elsif options[:all] Docs.all_versions else find_docs(names) end assert_docs(docs) download_docs(docs) generate_manifest puts 'Done' rescue Docs::DocNotFound => error handle_doc_not_found_error(error) end desc 'package ...', 'Package documentations' def package(*names) require 'unix_utils' docs = find_docs(names) assert_docs(docs) docs.each(&method(:package_doc)) puts 'Done' rescue Docs::DocNotFound => error handle_doc_not_found_error(error) end desc 'clean', 'Delete documentation packages' def clean File.delete(*Dir[File.join Docs.store_path, '*.tar.gz']) puts 'Done' end desc 'upload', '[private]' option :dryrun, type: :boolean def upload(*names) docs = find_docs(names) assert_docs(docs) docs.each do |doc| puts "Syncing #{doc.path}..." cmd = "aws s3 sync #{File.join(Docs.store_path, doc.path)} s3://docs.devdocs.io/#{doc.path} --delete" cmd << ' --dryrun' if options[:dryrun] system(cmd) end end private def find_docs(names) names.map do |name| name, version = name.split('@') Docs.find(name, version) end end def assert_docs(docs) if docs.empty? puts 'ERROR: called with no arguments.' puts 'Run "thor list" for usage patterns.' exit end end def handle_doc_not_found_error(error) puts %(ERROR: #{error}.) puts 'Run "thor docs:list" to see the list of docs and versions.' end def download_docs(docs) # Don't allow downloaded files to be created as StringIO require 'open-uri' OpenURI::Buffer.send :remove_const, 'StringMax' if OpenURI::Buffer.const_defined?('StringMax') OpenURI::Buffer.const_set 'StringMax', 0 require 'thread' length = docs.length mutex = Mutex.new i = 0 (1..4).map do Thread.new do while doc = docs.shift status = begin download_doc(doc) 'OK' rescue => e "FAILED (#{e.class}: #{e.message})" end mutex.synchronize { puts "(#{i += 1}/#{length}) #{doc.name}#{ " #{doc.version}" if doc.version} #{status}" } end end end.map(&:join) end def download_doc(doc) target = File.join(Docs.store_path, "#{doc.path}.tar.gz") open "http://dl.devdocs.io/#{doc.path}.tar.gz" do |file| FileUtils.mkpath(Docs.store_path) FileUtils.mv(file, target) unpackage_doc(doc) end end def unpackage_doc(doc) path = File.join(Docs.store_path, doc.path) FileUtils.mkpath(path) tar = UnixUtils.gunzip("#{path}.tar.gz") dir = UnixUtils.untar(tar) FileUtils.rm_rf(path) FileUtils.mv(dir, path) FileUtils.rm(tar) FileUtils.rm("#{path}.tar.gz") end def package_doc(doc) path = File.join Docs.store_path, doc.path if File.exist?(path) tar = UnixUtils.tar(path) gzip = UnixUtils.gzip(tar) FileUtils.mv(gzip, "#{path}.tar.gz") FileUtils.rm(tar) else puts %(ERROR: can't find "#{doc.name}" documentation files.) end end def generate_manifest Docs.generate_manifest end end