class Protobuf::Generators::FieldGenerator

def applicable_options

def applicable_options
  # Note on the strange use of `#inspect`:
  #   :boom.inspect #=> ":boom"
  #   :".boom.foo".inspect #=> ":\".boom.foo\""
  # An alternative to `#inspect` would be always adding double quotes,
  # but the generatated code looks un-idiomatic:
  #   ":\"#{:boom}\"" #=> ":\"boom\"" <-- Note the unnecessary double quotes
  @applicable_options ||= field_options.map { |k, v| "#{k.inspect} => #{v}" }
end

def compile

def compile
  run_once(:compile) do
    field_definition = if map?
                         ["map #{map_key_type_name}", map_value_type_name, name, number, applicable_options]
                       else
                         ["#{label} #{type_name}", name, number, applicable_options]
                       end
    puts field_definition.flatten.compact.join(', ')
  end
end

def default_value

def default_value
  @default_value ||= begin
                       if defaulted?
                         case descriptor.type.name
                         when :TYPE_ENUM
                           enum_default_value
                         when :TYPE_STRING, :TYPE_BYTES
                           string_default_value
                         when :TYPE_FLOAT, :TYPE_DOUBLE
                           float_double_default_value
                         else
                           verbatim_default_value
                         end
                       end
                     end
end

def defaulted?

def defaulted?
  descriptor.respond_to_has_and_present?(:default_value)
end

def deprecated?

def deprecated?
  descriptor.options.try(:deprecated?) { false }
end

def determine_map_entry

def determine_map_entry
  return nil if @enclosing_msg_descriptor.nil?
  return nil unless descriptor.label.name == :LABEL_REPEATED && descriptor.type.name == :TYPE_MESSAGE
  # find nested message type
  name_parts = descriptor.type_name.split(".")
  return nil if name_parts.size < 2 || name_parts[-2] != @enclosing_msg_descriptor.name
  nested = @enclosing_msg_descriptor.nested_type.find { |e| e.name == name_parts[-1] }
  return nested if !nested.nil? && nested.options.try(:map_entry?)
  nil
end

def determine_type_name(descriptor)

def determine_type_name(descriptor)
  case descriptor.type.name
  when :TYPE_MESSAGE, :TYPE_ENUM, :TYPE_GROUP then
    modulize(descriptor.type_name)
  else
    type_name = descriptor.type.name.to_s.downcase.sub(/^type_/, '')
    ":#{type_name}"
  end
end

def enum_default_value

def enum_default_value
  optionally_upcased_default =
    if ENV.key?('PB_UPCASE_ENUMS')
      verbatim_default_value.upcase
    elsif ENV.key?('PB_CAPITALIZE_ENUMS')
      verbatim_default_value.capitalize
    else
      verbatim_default_value
    end
  "#{type_name}::#{optionally_upcased_default}"
end

def extension?

def extension?
  descriptor.respond_to_has_and_present?(:extendee)
end

def field_options

def field_options
  @field_options ||= begin
                       opts = {}
                       opts[:default] = default_value if defaulted?
                       opts[:packed] = 'true' if packed?
                       opts[:deprecated] = 'true' if deprecated?
                       opts[:extension] = 'true' if extension?
                       if descriptor.options
                         descriptor.options.each_field do |field_option|
                           next unless descriptor.options.field?(field_option.name)
                           option_value = descriptor.options[field_option.name]
                           opts[field_option.fully_qualified_name] = serialize_value(option_value)
                         end
                       end
                       opts
                     end
end

def float_double_default_value

def float_double_default_value
  case verbatim_default_value
  when PROTO_INFINITY_DEFAULT then
    RUBY_INFINITY_DEFAULT
  when PROTO_NEGATIVE_INFINITY_DEFAULT then
    RUBY_NEGATIVE_INFINITY_DEFAULT
  when PROTO_NAN_DEFAULT then
    RUBY_NAN_DEFAULT
  else
    verbatim_default_value
  end
end

def initialize(field_descriptor, enclosing_msg_descriptor, indent_level)

def initialize(field_descriptor, enclosing_msg_descriptor, indent_level)
  super(field_descriptor, indent_level)
  @enclosing_msg_descriptor = enclosing_msg_descriptor
end

def label

def label
  @label ||= descriptor.label.name.to_s.downcase.sub(/label_/, '') # required, optional, repeated
end

def map?

def map?
  !map_entry.nil?
end

def map_entry

a map field.
represents the entries in the map. Returns nil if this field is not
If this field is a map field, this returns a message descriptor that
def map_entry
  @map_entry ||= determine_map_entry
end

def map_key_type_name

def map_key_type_name
  return nil if map_entry.nil?
  determine_type_name(map_entry.field.find { |v| v.name == 'key' && v.number == 1 })
end

def map_value_type_name

def map_value_type_name
  return nil if map_entry.nil?
  determine_type_name(map_entry.field.find { |v| v.name == 'value' && v.number == 2 })
end

def name

def name
  @name ||= descriptor.name.to_sym.inspect
end

def number

def number
  @number ||= descriptor.number
end

def packed?

def packed?
  descriptor.options.try(:packed?) { false }
end

def string_default_value

def string_default_value
  %("#{verbatim_default_value.gsub(/'/, '\\\\\'')}")
end

def type_name

Determine the field type
def type_name
  @type_name ||= determine_type_name(descriptor)
end

def verbatim_default_value

def verbatim_default_value
  descriptor.default_value
end