Class: ContractHandler
- Inherits:
-
YARD::Handlers::Ruby::Base
- Object
- YARD::Handlers::Ruby::Base
- ContractHandler
- Defined in:
- lib/yard-contracts/contract_handler.rb
Overview
Run the plugin handler by supplying it to yard with the –plugin flag
Instance Method Summary collapse
- #merge_params(doc, params) ⇒ Object
- #merge_return(tag, ret) ⇒ Object
- #new_params(doc, params) ⇒ Object
- #new_return(doc, ret) ⇒ Object
- #process ⇒ Object
- #process_params(doc, params) ⇒ Object
- #process_return(doc, ret) ⇒ Object
- #set_tag(tag, type, to_s) ⇒ Object
- #tag_text(to_s, text) ⇒ Object
Instance Method Details
#merge_params(doc, params) ⇒ Object
64 65 66 67 68 69 70 71 |
# File 'lib/yard-contracts/contract_handler.rb', line 64 def merge_params(doc, params) # Merge params into provided docstring otherwise there can be duplicates doc.(:param).each do |tag| next unless (param = params.find { |t| t[0].to_s == tag.name.to_s }) params.delete(param) set_tag(tag, param[1], param[2]) end end |
#merge_return(tag, ret) ⇒ Object
92 93 94 |
# File 'lib/yard-contracts/contract_handler.rb', line 92 def merge_return(tag, ret) set_tag(tag, ret[0], ret[1]) end |
#new_params(doc, params) ⇒ Object
73 74 75 76 77 78 79 80 |
# File 'lib/yard-contracts/contract_handler.rb', line 73 def new_params(doc, params) # If the docstring didn't contain all of the params already add the rest params.each do |param| doc.add_tag( YARD::Tags::Tag.new(:param, param[2].to_s, param[1].inspect, param[0]) ) end end |
#new_return(doc, ret) ⇒ Object
96 97 98 99 100 |
# File 'lib/yard-contracts/contract_handler.rb', line 96 def new_return(doc, ret) doc.add_tag( YARD::Tags::Tag.new(:return, ret[1].to_s, ret[0].inspect) ) end |
#process ⇒ Object
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 |
# File 'lib/yard-contracts/contract_handler.rb', line 19 def process # statement is a YARD attribute, subclassing YARD::Parser::Ruby::AstNode # Here it's class will be YARD::Parser::Ruby::MethodCallNode # MethodCallNode#line_range returns the lines the method call was over # AstNode#line gives the first line of the node # AstNode#traverse takes a block and yields child nodes # AstNode#jump returns the first node matching type, otherwise returns self # Go up the tree to namespace level, then jump to next def statement # Note: this won't document dynamicly defined methods. parent = statement.parent contract_last_line = statement.line_range.last # YARD::Parser::Ruby::MethodDefinitionNode def_method_ast = parent.traverse do |node| # Find the first def statement that comes after the contract we're on break node if node.line > contract_last_line && node.def? end ## Hacky way to test for class methods ## TODO: What about module methods? Probably broken. scope = def_method_ast.source.match(/ self\./) ? :class : :instance name = def_method_ast.method_name true params = def_method_ast.parameters # YARD::Parser::Ruby::ParameterNode contracts = statement.parameters # YARD::Parser::Ruby::AstNode ret = Contracts::Formatters::ParamContracts.new(params, contracts).return params = Contracts::Formatters::ParamContracts.new(params, contracts).params doc = YARD::DocstringParser.new.parse(statement.docstring).to_docstring process_params(doc, params) process_return(doc, ret) # YARD hasn't got to the def method yet, so we create a stub of it with # our docstring, when YARD gets to it properly it will fill in the rest. YARD::CodeObjects::MethodObject.new(namespace, name, scope) do |o| o.docstring = doc end # No `register()` it breaks stuff! Above implicit return value is enough. end |
#process_params(doc, params) ⇒ Object
59 60 61 62 |
# File 'lib/yard-contracts/contract_handler.rb', line 59 def process_params(doc, params) merge_params(doc, params) new_params(doc, params) end |
#process_return(doc, ret) ⇒ Object
82 83 84 85 86 87 88 89 90 |
# File 'lib/yard-contracts/contract_handler.rb', line 82 def process_return(doc, ret) if (tag = doc.tag :return) # Merge return into provided docstring otherwise there can be a duplicate merge_return(tag, ret) else # If the docstring didn't contain a return already add it new_return(doc, ret) end end |
#set_tag(tag, type, to_s) ⇒ Object
102 103 104 105 106 |
# File 'lib/yard-contracts/contract_handler.rb', line 102 def set_tag(tag, type, to_s) tag.types ||= [] tag.types << type.inspect tag.text = tag_text(to_s, tag.text) end |
#tag_text(to_s, text) ⇒ Object
108 109 110 |
# File 'lib/yard-contracts/contract_handler.rb', line 108 def tag_text(to_s, text) "#{to_s.empty? ? '' : "#{to_s}. "}#{text}" end |