| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117 |
- # encoding: UTF-8
- require 'thor'
- require 'pathname'
- require 'active_support/core_ext/hash/deep_merge'
- require 'active_support/inflector'
- require 'multi_json'
- require 'coderay'
- require 'pry'
- module Elasticsearch
- module API
- # A command line application based on [Thor](https://github.com/wycats/thor),
- # which will read the JSON API spec file(s), and generate
- # the Ruby source code (one file per API endpoint) with correct
- # module namespace, method names, and RDoc documentation,
- # as well as test files for each endpoint.
- #
- # Currently it only generates Ruby source, but can easily be
- # extended and adapted to generate source code for other
- # programming languages.
- #
- class SourceGenerator < Thor
- namespace 'api:code'
- include Thor::Actions
- __root = Pathname( File.expand_path('../../..', __FILE__) )
- desc "generate <PATH TO JSON SPEC FILES>", "Generate source code and tests from the REST API JSON specification"
- method_option :language, default: 'ruby', desc: 'Programming language'
- method_option :force, type: :boolean, default: false, desc: 'Overwrite the output'
- method_option :verbose, type: :boolean, default: false, desc: 'Output more information'
- method_option :input, default: File.expand_path('../../../api-spec', __FILE__), desc: 'Path to directory with JSON API specs'
- method_option :output, default: File.expand_path('../../../tmp/out', __FILE__), desc: 'Path to output directory'
- def generate(*files)
- self.class.source_root File.expand_path('../', __FILE__)
- @input = Pathname(options[:input])
- @output = Pathname(options[:output])
- # -- Test helper
- copy_file "templates/ruby/test_helper.rb", @output.join('test').join('test_helper.rb') if options[:language] == 'ruby'
- files.each do |file|
- @path = Pathname(file)
- @json = MultiJson.load( File.read(@path) )
- @spec = @json.values.first
- say_status 'json', @path, :yellow
- # say_status 'JSON', @spec.inspect, options[:verbose]
- @full_namespace = @json.keys.first.split('.')
- @namespace_depth = @full_namespace.size > 0 ? @full_namespace.size-1 : 0
- @module_namespace = @full_namespace[0, @namespace_depth]
- @method_name = @full_namespace.last
- # -- Ruby files
- @path_to_file = @output.join('api').join( @module_namespace.join('/') ).join("#{@method_name}.rb")
- empty_directory @output.join('api').join( @module_namespace.join('/') )
- template "templates/#{options[:language]}/method.erb", @path_to_file
- if options[:verbose]
- colorized_output = CodeRay.scan_file(@path_to_file, :ruby).terminal
- lines = colorized_output.split("\n")
- say_status options[:language].downcase,
- lines.first + "\n" + lines[1, lines.size].map { |l| ' '*14 + l }.join("\n"),
- :yellow
- end
- # --- Test files
- @test_directory = @output.join('test/api').join( @module_namespace.join('/') )
- @test_file = @test_directory.join("#{@method_name}_test.rb")
- empty_directory @test_directory
- template "templates/#{options[:language]}/test.erb", @test_file
- if options[:verbose]
- colorized_output = colorized_output = CodeRay.scan_file(@test_file, :ruby).terminal
- lines = colorized_output.split("\n")
- say_status options[:language].downcase,
- lines.first + "\n" + lines[1, lines.size].map { |l| ' '*14 + l }.join("\n"),
- :yellow
- say '▬'*terminal_width
- end
- end
- # -- Tree output
- if options[:verbose] && `which tree > /dev/null 2>&1`
- lines = `tree #{@output}`.split("\n")
- say_status 'tree',
- lines.first + "\n" + lines[1, lines.size].map { |l| ' '*14 + l }.join("\n")
- end
- end
- private
- # Create the hierarchy of directories based on API namespaces
- #
- def __create_directories(key, value)
- unless value['documentation']
- empty_directory @output.join(key)
- create_directory_hierarchy *value.to_a.first
- end
- end
- end
- end
- end
|