require 'pathname' require 'thread_safe/cache' module Dry module Validation module Messages class Abstract extend Dry::Configurable include Dry::Equalizer(:config) DEFAULT_PATH = Pathname(__dir__).join('../../../../config/errors.yml').realpath.freeze setting :paths, [DEFAULT_PATH] setting :root, 'errors'.freeze setting :lookup_options, [:root, :predicate, :rule, :val_type, :arg_type].freeze setting :lookup_paths, %w( %{root}.rules.%{rule}.%{predicate}.arg.%{arg_type} %{root}.rules.%{rule}.%{predicate} %{root}.%{predicate}.value.%{val_type}.arg.%{arg_type} %{root}.%{predicate}.value.%{val_type} %{root}.%{predicate}.arg.%{arg_type} %{root}.%{predicate} ).freeze setting :arg_type_default, 'default'.freeze setting :val_type_default, 'default'.freeze setting :arg_types, Hash.new { |*| config.arg_type_default }.update( Range => 'range' ) setting :val_types, Hash.new { |*| config.val_type_default }.update( Range => 'range', String => 'string' ) def self.cache @cache ||= ThreadSafe::Cache.new { |h, k| h[k] = ThreadSafe::Cache.new } end attr_reader :config def initialize @config = self.class.config end def rule(name, options = {}) path = "%{locale}.rules.#{name}" get(path, options) if key?(path, options) end def call(*args) cache.fetch_or_store(args.hash) do path, opts = lookup(*args) get(path, opts) if path end end alias_method :[], :call def lookup(predicate, options = {}) tokens = options.merge( root: root, predicate: predicate, arg_type: config.arg_types[options[:arg_type]], val_type: config.val_types[options[:val_type]] ) tokens[:rule] = predicate unless tokens.key?(:rule) opts = options.reject { |k, _| config.lookup_options.include?(k) } path = lookup_paths(tokens).detect do |key| key?(key, opts) && get(key, opts).is_a?(String) end [path, opts] end def lookup_paths(tokens) config.lookup_paths.map { |path| path % tokens } end def namespaced(namespace) Messages::Namespaced.new(namespace, self) end def root config.root end def cache self.class.cache[self] end end end end end