module DingSDK::Utils

def self._get_deep_object_query_params(metadata, field_name, obj)

def self._get_deep_object_query_params(metadata, field_name, obj)
  params = {}
  return params if obj.nil?
  if obj.respond_to? :fields
    obj_fields = obj.fields
    obj_fields.each do |obj_field|
      obj_param_metadata = obj_field.metadata[:query_param]
      next if obj_param_metadata.nil?
      val = obj.send(obj_field.name)
      next if val.nil?
      key = "#{metadata.fetch(:field_name, field_name)}[#{obj_param_metadata.fetch(:field_name, obj_field.name)}]"
      if val.is_a? Array
        val.each do |v|
          next if v.nil?
          params[key] = [] if !params.include? key
          params[key] << val_to_string(v)
        end
      else
        params[key] = [val_to_string(val)]
      end
    end
  elsif obj.is_a? Hash
    obj.each do |key, value|
      next if value.nil?
      param_key = "#{metadata.fetch(:field_name, field_name)}[#{key}]"
      if value.is_a? Array
        value.each do |val|
          next if val.nil?
          params[param_key] = [] if !params.include? param_key
          params[param_key].append(val_to_string(val))
        end
      else
        params[param_key] = [val_to_string(value)]
      end
    end
  end
  params
end

def self._get_delimited_query_params(metadata, field_name, obj, delimiter)

def self._get_delimited_query_params(metadata, field_name, obj, delimiter)
  get_query_param_field_name = lambda do |obj_field|
    obj_param_metadata = obj_field.metadata[:query_param]
    return '' if obj_param_metadata.nil?
    return obj_param_metadata.fetch(:field_name, obj_field.name)
  end
  _populate_form(field_name, metadata.fetch(:explode, true), obj, delimiter, &get_query_param_field_name)
end

def self._get_serialized_params(metadata, field_name, obj)

def self._get_serialized_params(metadata, field_name, obj)
  params = {}
  serialization = metadata.fetch(:serialization, '')
  params[metadata.fetch(:field_name, field_name)] = obj.marshal_json if serialization == 'json'
  params
end

def self._parse_basic_auth_scheme(req, scheme)

def self._parse_basic_auth_scheme(req, scheme)
  username, password = ''
  scheme_fields = scheme.fields
  scheme_fields.each do |scheme_field|
    metadata = scheme_field.metadata[:security]
    next if metadata.nil? || !metadata.include?(:field_name)
    field_name = metadata[:field_name]
    value = scheme.send(scheme_field.name)
    username = value if field_name == 'username'
    password = value if field_name == 'password'
  end
  data = "#{username}:#{password}".encode
  # Use strict_encode, because encode adds newlines after 60 chars
  # https://docs.ruby-lang.org/en/3.0/Base64.html#method-i-encode64
  req.headers['Authorization'] = "Basic #{Base64.strict_encode64(data)}"
end

def self._parse_security_option(req, option)

def self._parse_security_option(req, option)
  opt_fields = option.fields
  opt_fields.each do |opt_field|
    metadata = opt_field.metadata[:security]
    next if metadata.nil? || !metadata.include?(:scheme)
    _parse_security_scheme(req, metadata, option.send(opt_field.name))
  end
end

def self._parse_security_scheme(req, scheme_metadata, scheme)

def self._parse_security_scheme(req, scheme_metadata, scheme)
  scheme_type = scheme_metadata[:type]
  sub_type = scheme_metadata[:sub_type]
  if scheme.respond_to? :fields
    if scheme_type == 'http' && sub_type == 'basic'
      _parse_basic_auth_scheme(req, scheme)
      return
    end
    scheme_fields = scheme.fields
    scheme_fields.each do |field|
      metadata = field.metadata[:security]
      next if metadata.nil? || metadata[:field_name].nil?
      value = scheme.send(field.name)
      _parse_security_scheme_value(req, scheme_metadata, metadata, value)
    end
  else
    _parse_security_scheme_value(req, scheme_metadata, scheme_metadata, scheme)
  end
end

def self._parse_security_scheme_value(req, scheme_metadata, security_metadata, value)

def self._parse_security_scheme_value(req, scheme_metadata, security_metadata, value)
  scheme_type = scheme_metadata[:type]
  sub_type = scheme_metadata[:sub_type]
  header_name = security_metadata[:field_name]
  case scheme_type
  when 'apiKey'
    case sub_type
    when 'header'
      req.headers[header_name] = value
    when 'query'
      req.params[header_name] = value
    when 'cookie'
      req.cookies[header_name] = value
    else
      raise StandardError, 'not supported'
    end
  when 'openIdConnect'
    req.headers[header_name] = value.downcase.start_with?('bearer ') ? value : "Bearer #{value}"
  when 'oauth2'
    req.headers[header_name] = value.downcase.start_with?('bearer ') ? value : "Bearer #{value}"
  when 'http'
    if sub_type == 'bearer'
      req.headers[header_name] = value.downcase.start_with?('bearer ') ? value : "Bearer #{value}"
    else
      raise StandardError, 'not supported'
    end
  else
    raise StandardError, 'not supported'
  end
end

def self._populate_form(field_name, explode, obj, delimiter, &get_field_name_lambda)

def self._populate_form(field_name, explode, obj, delimiter, &get_field_name_lambda)
  params = {}
  return params if obj.nil?
  if obj.respond_to? :fields
    items = []
    obj_fields = obj.fields
    obj_fields.each do |obj_field|
      obj_field_name = get_field_name_lambda.call(obj_field)
      next if obj_field_name == ''
      val = obj.send(obj_field.name.to_sym)
      next if val.nil?
      if explode
        params[obj_field_name] = [val_to_string(val)]
      else
        items.append("#{obj_field_name}#{delimiter}#{val_to_string(val)}")
      end
    end
    params[field_name] = [items.join(delimiter)] if !items.empty?
  elsif obj.is_a? Hash
    items = []
    obj.each do |key, value|
      next if value.nil?
      if explode
        params[key] = val_to_string(value)
      else
        items.append("#{key}#{delimiter}#{val_to_string(value)}")
      end
    end
    params[field_name] = [items.join(delimiter)] if !items.empty?
  elsif obj.is_a? Array
    items = []
    obj.each do |value|
      next if value.nil?
      if explode
        params[field_name] = [] if !params.key? field_name
        params[field_name].append(val_to_string(value))
      else
        items.append(val_to_string(value))
      end
    end
    params[field_name] = items.map(&:to_s).join(delimiter) if !items.empty?
  else
    params[field_name] = val_to_string(obj)
  end
  params
end

def self._populate_from_globals(param_name, value, param_type, gbls)

def self._populate_from_globals(param_name, value, param_type, gbls)
  if value.nil? && !gbls.nil?
    global_value = gbls.dig(:parameters, param_type.to_sym, param_name.to_sym)
    value = global_value if !global_value.nil?
  end
  value
end

def self._serialize_header(explode, obj)

def self._serialize_header(explode, obj)
  return '' if obj.nil?
  if obj.respond_to? :fields
    items = []
    obj_fields = obj.fields
    obj_fields.each do |obj_field|
      obj_param_metadata = obj_field.metadata[:header]
      next if obj_param_metadata.nil?
      obj_field_name = obj_param_metadata.fetch(:field_name, obj_field.name)
      next if obj_field_name == ''
      val = obj.send(obj_field.name)
      next if val.nil?
      if explode
        items.append("#{obj_field_name}=#{val_to_string(val)}")
      else
        items.append(obj_field_name)
        items.append(val_to_string(val))
      end
    end
    items.join(',') if !items.empty?
  elsif obj.is_a? Hash
    items = []
    obj.each do |key, value|
      next if value.nil?
      if explode
        items.append("#{key}=#{val_to_string(value)}")
      else
        items.append(key)
        items.append(val_to_string(value))
      end
    end
    items.join(',') if !items.empty?
  elsif obj.is_a? Array
    items = obj.filter { |v| !v.nil? }.map { |v| val_to_string(v) }.join(',')
  else
    val_to_string(obj)
  end
end

def self.configure_request_security(req, security)

def self.configure_request_security(req, security)
  sec_fields = security.fields
  sec_fields.each do |sec_field|
    value = security.send(sec_field.name)
    next if value.nil?
    metadata = sec_field.metadata[:security]
    next if metadata.nil?
    _parse_security_option(req, value) if metadata[:option]
    if metadata[:scheme]
      # Special case for basic auth which could be a flattened struct
      if metadata[:sub_type] == 'basic' && !value.respond_to?(:fields)
        _parse_security_scheme(req, metadata, security)
      else
        _parse_security_scheme(req, metadata, value)
      end
    end
  end
end

def self.date_from_iso_format(optional)

def self.date_from_iso_format(optional)
  lambda do |s|
    return nil if optional && s.nil?
    return Date.iso8601(s)
  end
end

def self.datetime_from_iso_format(optional)

def self.datetime_from_iso_format(optional)
  lambda do |s|
    return nil if optional && s.nil?
    return DateTime.strptime(s, '%Y-%m-%dT%H:%M:%S.%NZ')
  end
end

def self.encode_form(form)

def self.encode_form(form)
  payload = {}
  form.each do |field_name, field|
    if field.length == 2
      if field[0].nil?
        payload[field_name] = field[1]
      else
        payload[field_name] = Faraday::Multipart::FilePart.new(field[0], '', field[1])
      end
    elsif field.length == 3
      payload[field_name] = Faraday::Multipart::ParamPart.new(field[1].to_json, field[2])
    end
  end
  payload
end

def self.enum_from_string(enum_type, optional)

def self.enum_from_string(enum_type, optional)
  lambda do |s|
    return nil if optional && s.nil?
    return enum_type.deserialize(s)
  end
end

def self.field_name(name)

def self.field_name(name)
  proc { |_, field_name = name| field_name }
end

def self.generate_url(clazz, server_url, path, path_params, gbls = nil)

def self.generate_url(clazz, server_url, path, path_params, gbls = nil)
  clazz.fields.each do |f|
    param_metadata = f.metadata[:path_param]
    next if param_metadata.nil?
    if param_metadata.fetch(:style, 'simple') == 'simple'
      param = path_params.send(f.name) if !path_params.nil?
      param = _populate_from_globals(f.name, param, 'pathParam', gbls)
    end
    f_name = param_metadata.fetch(:field_name, f.name)
    serialization = param_metadata.fetch(:serialization, '')
    if serialization != ''
      serialized_params = _get_serialized_params(param_metadata, f_name, param)
      serialized_params.each do |k, v|
        path = path.sub("{#{k}}", v)
      end
    else
      if param.is_a? Array
        pp_vals = []
        param.each do |pp_val|
          pp_vals.append(pp_val.to_s)
        end
        path = path.sub("{#{param_metadata.fetch(:field_name, f.name)}}", pp_vals.join(','))
      elsif param.is_a? Hash
        pp_vals = []
        param.each do |pp_key, pp_val|
          value = val_to_string(pp_val)
          if param_metadata.fetch(:explode, false)
            pp_vals.append("#{pp_key}=#{value}")
          else
            pp_vals.append("#{pp_key},#{value}")
          end
        end
        path = path.sub("{#{param_metadata.fetch(:field_name, f.name)}}", pp_vals.join(','))
      elsif !(param.is_a?(String) || param.is_a?(Integer) ||
        param.is_a?(Float) || param.is_a?(Complex) || param.is_a?(TrueClass) ||
        param.is_a?(FalseClass))
        pp_vals = []
        attrs = param.fields.filter { |field| field.name && param.respond_to?(field.name.to_sym) }.map(&:name)
        attrs.each do |attr|
          field = param.field(attr)
          param_value_metadata = field.metadata[:path_param]
          next if param_value_metadata.nil?
          parm_name = param_value_metadata.fetch(:field_name, f.name)
          param_field_val = param.send(attr)
          if param_field_val.is_a? T::Enum
            param_field_val = param_field_val.serialize
          elsif param_field_val.is_a? DateTime
            param_field_val = param_field_val.strftime('%Y-%m-%dT%H:%M:%S.%NZ')
          end
          if !field.nil? && T::Utils::Nilable.is_union_with_nilclass(field.type) && param_field_val.nil?
            next
          elsif param_metadata.fetch(:explode, false)
            pp_vals.append("#{parm_name}=#{param_field_val}")
          else
            pp_vals.append("#{parm_name},#{param_field_val}")
          end
        end
        path = path.sub("{#{param_metadata.fetch(:field_name, f.name)}}", pp_vals.join(','))
      else
        path = path.sub("{#{param_metadata.fetch(:field_name, f.name)}}", param.to_s)
      end
    end
  end
  server_url.delete_suffix('/') + path
end

def self.get_headers(headers_params, gbls = nil)

def self.get_headers(headers_params, gbls = nil)
  return {} if headers_params.nil?
  headers = {}
  param_fields = headers_params.fields
  param_fields.each do |f|
    metadata = f.metadata[:header]
    next if metadata.nil?
    value = _populate_from_globals(f.name, headers_params&.send(f.name), 'header', gbls)
    value = _serialize_header(metadata.fetch(:explode, false), value)
    headers[metadata.fetch(:field_name, f.name)] = value if !value.empty?
  end
  headers
end

def self.get_query_params(clazz, query_params, gbls = nil)

def self.get_query_params(clazz, query_params, gbls = nil)
  params = {}
  param_fields = clazz.fields
  param_fields.each do |f|
    request_metadata = f.metadata[:request]
    next if !request_metadata.nil?
    metadata = f.metadata[:query_param]
    next if metadata.nil?
    param_name = f.name
    value = query_params&.send(param_name.to_sym)
    value = _populate_from_globals(param_name, value, 'queryParam', gbls)
    f_name = metadata[:field_name]
    serialization = metadata.fetch(:serialization, '')
    if serialization != ''
      params = params.merge _get_serialized_params(
        metadata, f_name, value
      )
    else
      style = metadata.fetch(:style, 'form')
      case style
      when 'deepObject'
        params = params.merge _get_deep_object_query_params(
          metadata, f_name, value
        )
      when 'form'
        params = params.merge _get_delimited_query_params(
          metadata, f_name, value, ','
        )
      when 'pipeDelimited'
        params = params.merge _get_delimited_query_params(
          metadata, f_name, value, '|'
        )
      else
        raise StandardError, 'not yet implemented'
      end
    end
  end
  params
end

def self.marshal_json_complex(complex)

def self.marshal_json_complex(complex)
  if complex.is_a? Array
    complex.map { |v| Utils.marshal_json_complex(v) }.to_json
  elsif complex.is_a? Hash
    complex.transform_values { |v| Utils.marshal_json_complex(v) }.to_json
  elsif complex.respond_to? :marshal_json
    complex.marshal_json
  else
    complex.to_json
  end
end

def self.match_content_type(content_type, pattern)

def self.match_content_type(content_type, pattern)
  return true if content_type == pattern || ['*', '*/*'].include?(pattern)
  pieces = content_type.split(';')
  pieces.each do |piece|
    return true if pattern == piece.strip
  end
  false
end

def self.parse_field(field, data_class)

def self.parse_field(field, data_class)
  field_metadata = field.metadata[:metadata_string]
  return nil if field_metadata.nil?
  field_value = data_class.send(field.name)
  return nil if field_value.nil?
  field_value
end

def self.serialize_content_type(field_name, media_type, request)

def self.serialize_content_type(field_name, media_type, request)
  return media_type, marshal_json_complex(request), nil if media_type.match('(application|text)\/.*?\+*json.*')
  return serialize_multipart_form(media_type, request) if media_type.match('multipart\/.*')
  return media_type, serialize_form_data(field_name, request), nil if media_type.match('application\/x-www-form-urlencoded.*')
  return media_type, request, nil if request.is_a?(String) || request.is_a?(Array)
  raise StandardError, "invalid request body type #{type(request)} for mediaType {metadata['media_type']}"
end

def self.serialize_form_data(field_name, data)

def self.serialize_form_data(field_name, data)
  get_form_field_name = lambda do |obj_field|
    obj_param_metadata = obj_field.metadata[:form]
    return '' if obj_param_metadata.nil?
    return obj_param_metadata.fetch(:field_name, obj_field.name)
  end
  form = {}
  if data.respond_to? :fields
    data.fields.each do |field|
      val = data.send(field.name)
      next if val.nil?
      metadata = field.metadata[:form]
      next if metadata.nil?
      field_name = metadata.fetch(:field_name, field.name)
      if metadata[:json]
        form[field_name] = marshal_json_complex(val)
      else
        if metadata.fetch(:style, 'form') == 'form'
          form = form.merge(
            _populate_form(
              field_name, metadata.fetch(:explode, true), val, ',', &get_form_field_name
            )
          )
        else
          raise StandardError, "Invalid form style for field #{field.name}"
        end
      end
    end
  elsif data.is_a? Hash
    data.each do |key, value|
      form[key] = [val_to_string(value)]
    end
  else
    raise StandardError, "Invalid request body type for field #{field_name}"
  end
  form
end

def self.serialize_multipart_form(media_type, request)

def self.serialize_multipart_form(media_type, request)
  form = []
  request_fields = request.fields
  request_fields.each do |field|
    val = request.send(field.name)
    next if val.nil?
    field_metadata = field.metadata[:multipart_form]
    next if field_metadata.nil?
    if field_metadata[:file] == true
      file_fields = val.fields
      file_name = ''
      field_name = field_metadata[:field_name]
      content = nil
      file_fields.each do |file_field|
        file_metadata = file_field.metadata[:multipart_form]
        next if file_metadata.nil?
        if file_metadata[:content] == true
          content = val.send(file_field.name)
        else
          file_name = val.send(file_field.name)
        end
      end
      raise StandardError, 'invalid multipart/form-data file' if file_name == '' || content == nil?
      form.append([field_name, [file_name, content]])
    elsif field_metadata[:json] == true
      to_append = [
        field_metadata.fetch(:field_name, field.name), [
          nil, marshal_json_complex(val), 'application/json'
        ]
      ]
      form.append(to_append)
    else
      field_name = field_metadata.fetch(
        :field_name, field.name
      )
      if val.is_a? Array
        val.each do |value|
          next if value.nil?
          form.append(
            ["#{field_name}[]", [nil, val_to_string(value)]]
          )
        end
      else
        form.append([field_name, [nil, val_to_string(val)]])
      end
    end
  end
  [media_type, nil, form]
end

def self.serialize_request_body(request, request_field_name, serialization_method)

def self.serialize_request_body(request, request_field_name, serialization_method)
  return ['', nil, nil] if request.nil?
  return serialize_content_type(request_field_name, SERIALIZATION_METHOD_TO_CONTENT_TYPE[serialization_method], request) if !request.respond_to?(:fields) || !request.respond_to?(request_field_name)
  request_val = request.send(request_field_name)
  request_fields = request.fields
  request_metadata = nil
  request_fields.each do |f|
    if f.name == request_field_name
      request_metadata = f.metadata[:request]
      break
    end
  end
  raise StandardError, 'invalid request type' if request_metadata.nil?
  serialize_content_type(
    :request, request_metadata.fetch(:media_type, 'application/octet-stream'), request_val
  )
end

def self.template_url(url_with_params, params)

def self.template_url(url_with_params, params)
  params.each do |key, value|
    if value.respond_to? :serialize
      val_str = value.serialize
    else
      val_str = value
    end
    url_with_params = url_with_params.gsub("{#{key}}", val_str)
  end
  url_with_params
end

def self.unmarshal_complex(data, type)

def self.unmarshal_complex(data, type)
  begin
    value = unmarshal_json(JSON.parse(data), type)
  rescue TypeError, JSON::ParserError
    value = unmarshal_json(data, type)
  end
  value
end

def self.unmarshal_json(data, type)

def self.unmarshal_json(data, type)
  if T.simplifiable? type
    type = T.simplify_type type
  end
  if type.respond_to? :unmarshal_json
    type.unmarshal_json(data)
  elsif T.arr? type
    data.map { |v| Utils.unmarshal_complex(v, T.arr_of(type)) }
  elsif T.hash? type
    data.transform_values { |v| Utils.unmarshal_complex(v, T.hash_of(type)) }
  else
    data
  end
end

def self.val_to_string(val, primitives: true)

def self.val_to_string(val, primitives: true)
  if val.is_a? T::Enum
    val.serialize
  elsif val.is_a? DateTime
    val.strftime('%Y-%m-%dT%H:%M:%S.%NZ')
  elsif primitives
    val.to_s
  else
    val
  end
end