# frozen_string_literal: true require 'request_handler/schema_parser' require 'request_handler/error' module RequestHandler class FieldsetsParser def initialize(params:, allowed: {}, required: []) @params = params allowed.reject! { |_k, v| v == false } allowed.each_pair do |_key, value| raise InternalArgumentError, allowed: 'must be a Schema or a Boolean' unless RequestHandler.configuration.validation_engine.valid_schema?(value) || value.is_a?(TrueClass) || value.is_a?(FalseClass) end @allowed = allowed raise InternalArgumentError, required: 'must be an Array' unless required.is_a?(Array) @required = required end def run fields = params['fields'] raise_missing_fields_param unless fields fieldsets = fields.to_h.each_with_object({}) do |(type, values), memo| type = type.to_sym raise_invalid_field_option(type) memo[type] = parse_options(type, values) end check_required_fieldsets_types(fieldsets) end private def parse_options(type, values) values.split(',').map! do |option| parse_option(type, option) end end def parse_option(type, option) if allowed[type] == true option.to_sym else RequestHandler.configuration.validation_engine.validate!(option, allowed[type]).output.to_sym end rescue Validation::Error raise FieldsetsParamsError, [{ code: 'INVALID_QUERY_PARAMETER', status: '400', detail: "allowed fieldset does not include '#{option}'", source: { param: "fields[#{type}]" } }] end def check_required_fieldsets_types(fieldsets) missing = required - fieldsets.keys return fieldsets if missing.empty? raise_missing_fieldsets!(missing) end def raise_invalid_field_option(type) return if allowed.dig(type) raise OptionNotAllowedError, [ { code: 'INVALID_QUERY_PARAMETER', status: '400', detail: "fieldset for '#{type}' not allowed", source: { param: "fields[#{type}]" } } ] end def raise_missing_fields_param return if required.empty? raise_missing_fieldsets!(required) end def raise_missing_fieldsets!(missing) errors = missing.map do |type| { code: 'MISSING_QUERY_PARAMETER', status: '400', source: { param: '' }, detail: "missing required parameter fields[#{type}]" } end raise FieldsetsParamsError, errors end attr_reader :params, :allowed, :required end end