module GrapeDSL module Extend module Doc # helpers for doc generation def wiki_body(route,wrapper_begin,wrapper_end,wrapper_close) require 'grape-dsl/doc_mp' description_key= :body tmp_array= Array.new() params= nil evalue= nil content_type= nil #if route.route_path == "/booking/request(.:format)" # debugger #end case true when route.route_description.class <= ::String params= route.route_params when route.route_description.class <= ::Hash if !route.route_description[:content_type].nil? content_type= route.route_description[:content_type] end if description_key.nil? params= route.route_params evalue= "value[:type]" else params= route.route_description[description_key] evalue= "value" end when route.route_description.class <= ::NilClass params= route.route_params params ||= "no return" content_type= "TXT" else params= route.route_params content_type= "TXT" end case true when params.class <= ::Hash if params == route.route_params tmp_hash= Hash.new params.each do |key,value| tmp_hash[key]= value[:type].to_s end params= tmp_hash end params = params.convert_all_value_to_grape_dsl_format when params.class <= ::Class begin if params.to_s.include? '::' if params.to_s.downcase.include? 'boolean' params= params.to_s.split('::').last end end content_type= "TXT" end when params.class <= ::String content_type= "TXT" else begin params= "no params spec" content_type= "TXT" end end content_type ||= 'TXT' content_type = content_type.keys if content_type.class <= ::Hash [*content_type].each do |one_content_type| case one_content_type.to_s.downcase when 'json' begin tmp_array.push [wrapper_begin,one_content_type.to_s,wrapper_end].join require 'json' formatted_string= params.to_json { '{' => "{\n", '}' => "\n}", ',' => ",\n" }.each do |from,to| formatted_string.gsub!(from,to) end formatted_string.gsub!(/^"/," \"") tmp_array.push formatted_string tmp_array.push wrapper_close end when "txt" tmp_array.push(params.inspect) end end return tmp_array end # this method help create from grape params and description a ppt like redmine wiki doc # Usage: # ##> example variable for description (hash obj) #description= Hash.new # ##> optional -> :description #description[:desc]= String.new # ##> response body like if JSON than a ruby Hash obj with the Structure and the values are the types #description[:body]= Hash.new # ##> response body code type -> like json usually same as content_type (format :xy) #description[:content_type]= String.new #> "JSON" # #desc description #params do # optional :blabla, :type => String, :desc => "bla bla desc" # requires :xy, type: String, desc: "XY desc" #end #get "xy" do # #end # ##>--------------------------------------------------------------------------------------------------- #> OR the classic #desc "My awsome String Description" #params do # optional :blabla, :type => String, :desc => "bla bla desc" # requires :xy, type: String, desc: "XY desc" #end #delete "xy" do # #end # ##>--------------------------------------------------------------------- ## For the method use # ## for a targeted specified class #Grape.create_redmine_wiki_doc target_class: REST::API, # path: File.expand_path(File.join(File.dirname(__FILE__),"test_file.txt")) # ## for all grape subclass (directs and indirects) #Grape.create_redmine_wiki_doc path: File.expand_path(File.join(File.dirname(__FILE__),"test_file.txt")) # def create_wiki_doc(*args) # default set in args begin args= Hash[*args] args.dup.each do |key,value| if key.class != Symbol args[key.to_s.to_sym]= value args.delete key end end if args[:path].nil? raise ArgumentError,"You must set a file path with a file name in order to create documentation for grape!" end args[:desc_files] ||= Array.new [:desc,:desc_file,:extra_desc_file].each do |one_key| args[:desc_files] += args[(one_key.to_s+"s").to_sym] if args[(one_key.to_s+"s").to_sym].class == Array args[:desc_files].push(args[one_key]) if args[one_key].class == String end args[:type] ||= args[:doc_type] args[:type] ||= 'wiki' #args[:path], #args[:extra_desc_file] #args[:target_class] #args[:type] end # defaults begin uni_tab= "" case args[:type].to_s.downcase when "redmine","redmine_wiki","redmine-wiki","redminewiki" begin mid_tab= " "*3 bsym= "*" isym= "_" htsym= "* " mtsym= htsym[0]*2 +" " stsym= htsym[0]*3 +" " hheader= "h3. " mheader= "h4. " sheader= "h5. " container_markup_begin= "
"
                container_markup_close= "
" toc_mark= "\n{{>toc}}\n" end when "github","wiki","md" begin mid_tab= " "*3 bsym= "*" isym= "_" htsym= "* " mtsym= " * " stsym= " * " hheader= "## " mheader= "### " sheader= "#### " container_markup_begin= "```" container_markup_end= "" container_markup_close= "```" toc_mark= "" end else raise ArgumentError, "invalid :type has been set, try github or redmine" end end # site name begin write_out_array = Array.new write_out_array.push "#{hheader}#{$0} REST Interface Documentation\n\n" end # description begin args[:desc_files].each do |extra_desc_file_path| write_out_array.push "#{sheader}#{extra_desc_file_path.split(File::Separator).last.split('.')[0].camelcase}\n" write_out_array.push " "+File.open(extra_desc_file_path,"r").read+"\n" end end # table of contents begin write_out_array.push toc_mark end # classes array begin rest_models= Array.new end if args[:target_class].nil? Grape::API.each_subclass do |one_class| rest_models.push(one_class) end else if args[:target_class].class != Class && args[:target_class] != nil raise ArgumentError, "invalid input :target_class is not a Class obj" end rest_models.push(args[:target_class]) end rest_models.each do |rest_api_model| next if Grape::API == rest_api_model rest_api_model.routes.map do |route| method_name= "#{hheader}Request: #{route.route_path} call: #{route.route_method.to_s.downcase} part" # check that does the method already in the documentation unless write_out_array.include?(method_name) # create call name begin write_out_array.push method_name end # request begin # create request description begin write_out_array.push("\n"+(uni_tab*1)+"#{mheader}Request description") case true when route.route_description.class <= String route.route_description.each_line do |one_line| write_out_array.push((uni_tab*2)+htsym+one_line.chomp) end when route.route_description.class <= Hash begin description_msg = nil [:d,:desc,:description].each do |sym| description_msg ||= route.route_description[sym] end description_msg ||= "No description available for this path" description_msg= [*description_msg] description_msg.dup.each do |element| next unless element.include?("\n") index_n= description_msg.index(element) description_msg.delete(element) new_elements= element.split("\n") new_elements.size.times do |counter_n| description_msg.insert(index_n + counter_n - 1, new_elements[counter_n - 1] ) end end description_msg.each do |one_line| write_out_array.push((uni_tab*2)+htsym+one_line.chomp) end end end end # pre request begin write_out_array.push("\n#{mheader}request\n") end # create route method begin write_out_array.push((uni_tab*2)+"#{htsym}#{bsym}method:#{bsym}#{mid_tab} #{route.route_method}") end # create route path begin write_out_array.push((uni_tab*2)+"#{htsym}#{bsym}path:#{bsym}#{mid_tab} #{route.route_path}") end # create route content_type begin write_out_array.push((uni_tab*2)+"#{htsym}#{bsym}headers:#{bsym}#{mid_tab}") rest_api_model.content_types.each do |one_format_type,one_format_header| write_out_array.push "#{mtsym}#{uni_tab*2}#{one_format_header}" end write_out_array.push "" end # parameters begin new_docs_element= Array.new if route.route_params.count == 0 new_docs_element.push " No specified or special params" else new_docs_element.push "" new_docs_element.push "#{htsym}#{isym}#{bsym}Parameters#{bsym}#{isym}" route.route_params.each do |key,value| new_docs_element.push "#{mtsym}#{isym}#{key}#{isym}" value.each do |value_key,value_value| new_docs_element.push "#{stsym}#{value_key}: #{value_value}" end end new_docs_element.push "\n" end refactored_element= Array.new new_docs_element.each do |one_element| refactored_element.push((uni_tab*2)+one_element) end write_out_array.push refactored_element.join("\n") end end # response begin # pre response begin write_out_array.push("\n#{mheader}response\n") end #> TODO make better implementation for others to use #create route content_type begin if !Grape::Endpoint.config_obj.nil? write_out_array.push((uni_tab*2)+"#{sheader}Extra headers:") Grape::Endpoint.header_config_obj.each do |header_key,header_value| write_out_array.push "#{htsym}#{header_key}: #{header_value.join(', ')}" end write_out_array.push "" end end if Grape::Endpoint.respond_to?(:config_obj) && Grape::Endpoint.respond_to?(:header_config_obj) # create response bodies begin #TODO check out why not working normaly with evry path! write_out_array.push((uni_tab*2)+"#{sheader}*body:*") wiki_body(route,container_markup_begin,container_markup_end,container_markup_close ).each do |one_element| write_out_array.push one_element end write_out_array.push "" end end # error resp begin # pre error begin write_out_array.push("\n#{mheader}response in case of failure\n") end # create error response headers begin end # create error response bodies begin #write_out_array.push((uni_tab*2)+"*body:*") write_out_array.push((uni_tab*2)+"#{htsym}*Internal Server Error:500*") end end # after space begin write_out_array.push "\n----\n" end end end end File.new(args[:path],"w").write write_out_array.join("\n") return nil end alias :create_ppt_doc :create_wiki_doc end end end Grape.__send__ :extend, ::GrapeDSL::Extend::Doc