DannyBen/bashly

View on GitHub
lib/bashly/commands/render.rb

Summary

Maintainability
A
0 mins
Test Coverage
require 'filewatcher'
require 'tty-markdown'

module Bashly
  module Commands
    class Render < Base
      help 'Render the bashly data structure using custom templates'

      usage 'bashly render SOURCE TARGET [--watch --show PATH]'
      usage 'bashly render SOURCE --about'
      usage 'bashly render --list'
      usage 'bashly render (-h|--help)'

      param 'SOURCE', <<~HELP
        An ID to an internal templates source, or a path to a custom templates directory.

        A leading colon (:) denotes an internal ID (see `--list`).
      HELP

      param 'TARGET', 'Output directory'

      option '-w --watch', 'Watch bashly.yml and the templates source for changes and render on change'
      option '-s --show PATH', <<~USAGE
        After rendering, show the result generated in PATH.

        The provided PATH is treated as relative TARGET.

        Note that this works only if the template source supports it.
      USAGE

      option '-l --list', 'Show list of built-in templates'
      option '-a --about', 'Show information about a given templates source'

      example 'bashly render --list'
      example 'bashly render :markdown --about'
      example 'bashly render :markdown docs --watch'
      example 'bashly render :markdown docs --show "cli-download.1"'
      example 'bashly render /path/to/templates ./out_path'

      attr_reader :watching, :target, :source

      def run
        if args['--list'] then show_list
        elsif args['--about'] then show_about
        else
          start_render
        end
      end

    private

      def show_list
        RenderSource.internal.each_value do |source|
          say "g`:#{source.selector.to_s.ljust 10}`  #{source.summary}"
        end
      end

      def show_about
        puts TTY::Markdown.parse(render_source.readme)
      end

      def start_render
        @target = args['TARGET']
        @watching = args['--watch']

        render
        watch if watching
      end

      def render
        render_source.render target, show: args['--show']
      end

      def watch
        say "g`watching`\n"

        Filewatcher.new(watchables).watch do
          render
          say "g`waiting`\n"
        end
      end

      def render_source
        @render_source ||= begin
          source = RenderSource.new selector
          raise "Invalid render source: #{args['SOURCE']}" unless source.exist?

          source
        end
      end

      def selector
        return args['SOURCE'] unless args['SOURCE'].start_with? ':'

        args['SOURCE'][1..].to_sym
      end

      def watchables
        @watchables ||= [Settings.config_path, render_source.path]
      end
    end
  end
end