maestrano/maestrano-connector-rails

View on GitHub
app/controllers/maestrano/connec_controller.rb

Summary

Maintainability
A
3 hrs
Test Coverage
class Maestrano::ConnecController < Maestrano::Rails::WebHookController
  def notifications
    begin
      params.except(:tenant, :controller, :action, :connec).each do |entity_name, entities|
        entity_class_hash = find_entity_class(entity_name)
        next Maestrano::Connector::Rails::ConnectorLogger.log('info', nil, "Received notification from Connec! for unknow entity: #{entity_name}") unless entity_class_hash

        entities.each do |entity|
          begin
            organization = find_valid_organization(entity[:group_id], params[:tenant], entity_class_hash)
            next if organization.blank?

            Maestrano::Connector::Rails::ConnectorLogger.log('info', organization, "Processing entity from Connec! webhook, entity_name=\"#{entity_name}\", data=\"#{entity}\"")

            connec_client = Maestrano::Connector::Rails::ConnecHelper.get_client(organization)
            external_client = Maestrano::Connector::Rails::External.get_client(organization)
            last_synchronization_date = organization.last_synchronization_date

            entity_instance = entity_class_hash[:class].new(organization, connec_client, external_client, {})
            entity_instance.before_sync(last_synchronization_date)

            # Build expected input for consolidate_and_map_data
            mapped_entity = map_entity(entity_class_hash, entity_instance, entity_name, entity)

            entity_instance.push_entities_to_external(mapped_entity[:connec_entities])
            entity_instance.after_sync(last_synchronization_date)
          rescue => e
            Maestrano::Connector::Rails::ConnectorLogger.log('warn', organization, "error processing notification entity_name=\"#{entity_name}\", message=\"#{e.message}\", trace=\"#{e.backtrace}\"")
          end
        end
      end
    rescue => e
      Maestrano::Connector::Rails::ConnectorLogger.log('warn', nil, "error processing notification #{e.message} - #{e.backtrace}")
    end

    head 200, content_type: 'application/json'
  end

  private

    def find_valid_organization(uid, tenant, entity_class_hash)
      organization = Maestrano::Connector::Rails::Organization.find_by(uid: uid, tenant: tenant)

      if organization.nil?
        Maestrano::Connector::Rails::ConnectorLogger.log('info', organization, "Received notification from Connec! for an unknown organization, organization_uid=\"#{uid}\", tenant=\"#{tenant}\"")
        return nil
      end

      if organization.oauth_uid.blank?
        Maestrano::Connector::Rails::ConnectorLogger.log('info', organization, "Received notification from Connec! for an organization not linked, organization_uid=\"#{uid}\", tenant=\"#{tenant}\"")
        return nil
      end

      unless organization.sync_enabled && organization.synchronized_entities[entity_class_hash[:name].to_sym][:can_push_to_external]
        Maestrano::Connector::Rails::ConnectorLogger.log('info', organization, "Skipping notification from Connec! webhook, entity_name=\"#{entity_class_hash[:name]}\"")
        return nil
      end

      organization
    end

    def find_entity_class(entity_name)
      parametrised_entity_name = entity_name.parameterize('_').pluralize
      Maestrano::Connector::Rails::External.entities_list.each do |entity_name_from_list|
        clazz = "Entities::#{entity_name_from_list.singularize.titleize.split.join}".constantize
        if clazz.methods.include?('connec_entities_names'.to_sym)
          formatted_entities_names = clazz.connec_entities_names.map { |n| n.parameterize('_').pluralize }
          return {class: clazz, is_complex: true, name: entity_name_from_list} if formatted_entities_names.include?(parametrised_entity_name)
        elsif clazz.methods.include?('connec_entity_name'.to_sym) && clazz.normalized_connec_entity_name == parametrised_entity_name
          return {class: clazz, is_complex: false, name: entity_name_from_list}
        end
      end
      nil
    end

    def map_entity(entity_class_hash, entity_instance, entity_name, entity)
      # Build expected input for consolidate_and_map_data
      if entity_class_hash[:is_complex]
        connec_hash_of_entities = Maestrano::Connector::Rails::ComplexEntity.build_hash_with_entities(entity_instance.class.connec_entities_names, entity_name, ->(name) { name.parameterize('_').pluralize }, [entity])
        filtered_entities = entity_instance.filter_connec_entities(connec_hash_of_entities)

        empty_external_hash = Maestrano::Connector::Rails::ComplexEntity.build_empty_hash(entity_instance.class.external_entities_names)
        mapped_entity = entity_instance.consolidate_and_map_data(filtered_entities, empty_external_hash)
      else
        filtered_entities = entity_instance.filter_connec_entities([entity])
        mapped_entity = entity_instance.consolidate_and_map_data(filtered_entities, [])
      end

      mapped_entity
    end
end