# frozen_string_literal: true require "bootsnap/bootsnap" module Bootsnap module CompileCache module JSON class << self attr_accessor(:msgpack_factory, :supported_options) attr_reader(:cache_dir) def cache_dir=(cache_dir) @cache_dir = cache_dir.end_with?("/") ? "#{cache_dir}json" : "#{cache_dir}-json" end def input_to_storage(payload, _) obj = ::JSON.parse(payload) msgpack_factory.dump(obj) end def storage_to_output(data, kwargs) if kwargs&.key?(:symbolize_names) kwargs[:symbolize_keys] = kwargs.delete(:symbolize_names) end msgpack_factory.load(data, kwargs) end def input_to_output(data, kwargs) ::JSON.parse(data, **(kwargs || {})) end def precompile(path) Bootsnap::CompileCache::Native.precompile( cache_dir, path.to_s, self, ) end def install!(cache_dir) self.cache_dir = cache_dir init! if ::JSON.respond_to?(:load_file) ::JSON.singleton_class.prepend(Patch) end end def init! require "json" require "msgpack" self.msgpack_factory = MessagePack::Factory.new self.supported_options = [:symbolize_names] if supports_freeze? self.supported_options = [:freeze] end supported_options.freeze end private def supports_freeze? ::JSON.parse('["foo"]', freeze: true).first.frozen? && MessagePack.load(MessagePack.dump("foo"), freeze: true).frozen? end end module Patch def load_file(path, *args) return super if args.size > 1 if (kwargs = args.first) return super unless kwargs.is_a?(Hash) return super unless (kwargs.keys - ::Bootsnap::CompileCache::JSON.supported_options).empty? end ::Bootsnap::CompileCache::Native.fetch( Bootsnap::CompileCache::JSON.cache_dir, File.realpath(path), ::Bootsnap::CompileCache::JSON, kwargs, ) end ruby2_keywords :load_file if respond_to?(:ruby2_keywords, true) end end end end