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
|
# frozen_string_literal: true
module JMESPath
# @api private
module Nodes
class Field < Node
def initialize(key)
@key = key
@key_sym = key.respond_to?(:to_sym) ? key.to_sym : nil
end
def visit(value)
if value.respond_to?(:to_ary) && @key.is_a?(Integer)
value.to_ary[@key]
elsif value.respond_to?(:to_hash)
value = value.to_hash
if !(v = value[@key]).nil?
v
elsif @key_sym && !(v = value[@key_sym]).nil?
v
end
elsif value.is_a?(Struct) && value.respond_to?(@key)
value[@key]
end
end
def chains_with?(other)
other.is_a?(Field)
end
def chain(other)
ChainedField.new([@key, *other.keys])
end
protected
def keys
[@key]
end
end
class ChainedField < Field
def initialize(keys)
@keys = keys
@key_syms = keys.each_with_object({}) do |k, syms|
syms[k] = k.to_sym if k.respond_to?(:to_sym)
end
end
def visit(obj)
@keys.reduce(obj) do |value, key|
if value.respond_to?(:to_ary) && key.is_a?(Integer)
value.to_ary[key]
elsif value.respond_to?(:to_hash)
value = value.to_hash
if !(v = value[key]).nil?
v
elsif (sym = @key_syms[key]) && !(v = value[sym]).nil?
v
end
elsif value.is_a?(Struct) && value.respond_to?(key)
value[key]
end
end
end
def chain(other)
ChainedField.new([*@keys, *other.keys])
end
private
attr_reader :keys
end
end
end
|