-
-
Notifications
You must be signed in to change notification settings - Fork 399
/
attribute_handler.rb
87 lines (79 loc) · 2.8 KB
/
attribute_handler.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
# frozen_string_literal: true
# Handles +attr_*+ statements in modules/classes
class YARD::Handlers::Ruby::AttributeHandler < YARD::Handlers::Ruby::Base
handles method_call(:attr)
handles method_call(:attr_reader)
handles method_call(:attr_writer)
handles method_call(:attr_accessor)
namespace_only
process do
return if statement.type == :var_ref || statement.type == :vcall
read = true
write = false
params = statement.parameters(false).dup
# Change read/write based on attr_reader/writer/accessor
case statement.method_name(true)
when :attr
# In the case of 'attr', the second parameter (if given) isn't a symbol.
if params.size == 2
write = true if params.pop == s(:var_ref, s(:kw, "true"))
end
when :attr_accessor
write = true
when :attr_reader
# change nothing
when :attr_writer
read = false
write = true
end
# Add all attributes
validated_attribute_names(params).each do |name|
namespace.attributes[scope][name] ||= SymbolHash[:read => nil, :write => nil]
# Show their methods as well
{:read => name, :write => "#{name}="}.each do |type, meth|
if type == :read ? read : write
o = MethodObject.new(namespace, meth, scope)
if type == :write
o.parameters = [['value', nil]]
src = "def #{meth}(value)"
full_src = "#{src}\n @#{name} = value\nend"
doc = "Sets the attribute #{name}\n@param value the value to set the attribute #{name} to."
else
src = "def #{meth}"
full_src = "#{src}\n @#{name}\nend"
doc = "Returns the value of attribute #{name}."
end
o.source ||= full_src
o.signature ||= src
register(o)
o.docstring = doc if o.docstring.blank?(false)
# Register the object explicitly
namespace.attributes[scope][name][type] = o
else
obj = namespace.children.find {|other| other.name == meth.to_sym && other.scope == scope }
# register an existing method as attribute
namespace.attributes[scope][name][type] = obj if obj
end
end
end
end
protected
# Strips out any non-essential arguments from the attr statement.
#
# @param [Array<Parser::Ruby::AstNode>] params a list of the parameters
# in the attr call.
# @return [Array<String>] the validated attribute names
# @raise [Parser::UndocumentableError] if the arguments are not valid.
def validated_attribute_names(params)
params.map do |obj|
case obj.type
when :symbol_literal
obj.jump(:ident, :op, :kw, :const).source
when :string_literal
obj.jump(:string_content).source
else
raise YARD::Parser::UndocumentableError, obj.source
end
end
end
end