# frozen_string_literal: true require 'strscan' require 'multi_json' require 'jsonpath/proxy' require 'jsonpath/enumerable' require 'jsonpath/version' require 'jsonpath/parser' # JsonPath: initializes the class with a given JsonPath and parses that path # into a token array. class JsonPath PATH_ALL = '$..*'.freeze attr_accessor :path def initialize(path, opts = {}) @opts = opts scanner = StringScanner.new(path.strip) @path = [] until scanner.eos? if token = scanner.scan(/\$\B|@\B|\*|\.\./) @path << token elsif token = scanner.scan(/[\$@a-zA-Z0-9:{}_-]+/) @path << "['#{token}']" elsif token = scanner.scan(/'(.*?)'/) @path << "[#{token}]" elsif token = scanner.scan(/\[/) @path << find_matching_brackets(token, scanner) elsif token = scanner.scan(/\]/) raise ArgumentError, 'unmatched closing bracket' elsif scanner.scan(/\./) nil elsif token = scanner.scan(/[><=] \d+/) @path.last << token elsif token = scanner.scan(/./) begin @path.last << token rescue RuntimeError raise ArgumentError, "character '#{token}' not supported in query" end end end end def find_matching_brackets(token, scanner) count = 1 until count.zero? if t = scanner.scan(/\[/) token << t count += 1 elsif t = scanner.scan(/\]/) token << t count -= 1 elsif t = scanner.scan(/[^\[\]]+/) token << t elsif scanner.eos? raise ArgumentError, 'unclosed bracket' end end token end def join(join_path) res = deep_clone res.path += JsonPath.new(join_path).path res end def on(obj_or_str, opts = {}) a = enum_on(obj_or_str).to_a if opts[:symbolize_keys] a.map! do |e| e.each_with_object({}) { |(k, v), memo| memo[k.to_sym] = v; } end end a end def first(obj_or_str, *args) enum_on(obj_or_str).first(*args) end def enum_on(obj_or_str, mode = nil) JsonPath::Enumerable.new(self, self.class.process_object(obj_or_str), mode, @opts) end alias [] enum_on def self.on(obj_or_str, path, opts = {}) new(path, opts).on(process_object(obj_or_str)) end def self.for(obj_or_str) Proxy.new(process_object(obj_or_str)) end private def self.process_object(obj_or_str) obj_or_str.is_a?(String) ? MultiJson.decode(obj_or_str) : obj_or_str end def deep_clone Marshal.load Marshal.dump(self) end end