class SplitIoClient::SplitFactory

def build_fetchers

def build_fetchers
  @split_fetcher = SplitFetcher.new(@splits_repository, @rule_based_segment_repository, @api_key, @config, @runtime_producer)
  @segment_fetcher = SegmentFetcher.new(@segments_repository, @api_key, @config, @runtime_producer)
end

def build_flag_sets_filter

def build_flag_sets_filter
  @flag_sets_filter = SplitIoClient::Cache::Filter::FlagSetsFilter.new(@config.flag_sets_filter)
end

def build_impression_counter

def build_impression_counter
  @impression_counter = Engine::Common::ImpressionCounter.new
end

def build_impressions_components

def build_impressions_components
  build_impressions_observer
  build_impression_counter
  @impressions_manager = Engine::Common::ImpressionManager.new(@config, @impressions_repository, @impression_counter, @runtime_producer, @impression_observer, @unique_keys_tracker)
end

def build_impressions_observer

def build_impressions_observer
  if (@config.cache_adapter == :redis && @config.impressions_mode != :optimized)
    @impression_observer = Observers::NoopImpressionObserver.new
  else
    @impression_observer = Observers::ImpressionObserver.new
  end
end

def build_impressions_sender_adapter

def build_impressions_sender_adapter
  @impressions_api = Api::Impressions.new(@api_key, @config, @runtime_producer)
  @impressions_sender_adapter = Cache::Senders::ImpressionsSenderAdapter.new(@config, @telemetry_api, @impressions_api)
end

def build_repositories

def build_repositories
  if @config.cache_adapter.is_a? SplitIoClient::Cache::Adapters::RedisAdapter
    @flag_sets_repository = SplitIoClient::Cache::Repositories::RedisFlagSetsRepository.new(@config)
  else
    @flag_sets_repository = SplitIoClient::Cache::Repositories::MemoryFlagSetsRepository.new(@config.flag_sets_filter)
  end
  @splits_repository = SplitsRepository.new(@config, @flag_sets_repository, @flag_sets_filter)
  @segments_repository = SegmentsRepository.new(@config)
  @rule_based_segment_repository = RuleBasedSegmentsRepository.new(@config)
  @impressions_repository = ImpressionsRepository.new(@config)
  @events_repository = EventsRepository.new(@config, @api_key, @runtime_producer)
end

def build_streaming_components

def build_streaming_components
  @push_status_queue = Queue.new
  splits_worker = SSE::Workers::SplitsWorker.new(@synchronizer, @config, @splits_repository, @runtime_producer, @segment_fetcher, @rule_based_segment_repository)
  segments_worker = SSE::Workers::SegmentsWorker.new(@synchronizer, @config, @segments_repository)
  notification_manager_keeper = SSE::NotificationManagerKeeper.new(@config, @runtime_producer, @push_status_queue)
  notification_processor = SSE::NotificationProcessor.new(@config, splits_worker, segments_worker)
  event_parser = SSE::EventSource::EventParser.new(config)
  sse_client = SSE::EventSource::Client.new(@config, @api_key, @runtime_producer, event_parser, notification_manager_keeper, notification_processor, @push_status_queue)
  @sse_handler = SSE::SSEHandler.new(@config, splits_worker, segments_worker, sse_client)
  @push_manager = Engine::PushManager.new(@config, @sse_handler, @api_key, @runtime_producer)
end

def build_sync_manager

def build_sync_manager
  @sync_manager = Engine::SyncManager.new(@config, @synchronizer, @runtime_producer, @telemetry_synchronizer, @status_manager, @sse_handler, @push_manager, @push_status_queue)
end

def build_synchronizer

def build_synchronizer
  params = {
    split_fetcher: @split_fetcher,
    segment_fetcher: @segment_fetcher,
    imp_counter: @impression_counter,
    telemetry_synchronizer: @telemetry_synchronizer,
    impressions_sender_adapter: @impressions_sender_adapter,
    impressions_api: @impressions_api,
    unique_keys_tracker: @unique_keys_tracker
  }
  @synchronizer = Engine::Synchronizer.new(repositories, @config, params)
end

def build_telemetry_components

def build_telemetry_components
  @evaluation_consumer = Telemetry::EvaluationConsumer.new(@config)
  @evaluation_producer = Telemetry::EvaluationProducer.new(@config)
  @init_consumer = Telemetry::InitConsumer.new(@config)
  @init_producer = Telemetry::InitProducer.new(@config)
  @runtime_consumer = Telemetry::RuntimeConsumer.new(@config)
  @runtime_producer = Telemetry::RuntimeProducer.new(@config)
  @telemetry_consumers = { init: @init_consumer, evaluation: @evaluation_consumer, runtime: @runtime_consumer }
end

def build_telemetry_synchronizer(flag_sets, flag_sets_invalid)

def build_telemetry_synchronizer(flag_sets, flag_sets_invalid)
  @telemetry_api = Api::TelemetryApi.new(@config, @api_key, @runtime_producer)
  @telemetry_synchronizer = Telemetry::Synchronizer.new(@config, @telemetry_consumers, @init_producer, repositories, @telemetry_api, flag_sets, flag_sets_invalid)
end

def build_unique_keys_tracker

def build_unique_keys_tracker
  bf = Cache::Filter::BloomFilter.new(30_000_000)
  filter_adapter = Cache::Filter::FilterAdapter.new(@config, bf)
  cache = Concurrent::Hash.new
  @unique_keys_tracker = Engine::Impressions::UniqueKeysTracker.new(@config, filter_adapter, @impressions_sender_adapter, cache)
end

def initialize(api_key, config_hash = {})

def initialize(api_key, config_hash = {})
  at_exit do
    unless ENV['SPLITCLIENT_ENV'] == 'test'
      if (Process.pid == ROOT_PROCESS_ID)
        @config.logger.info('Split SDK shutdown started...')
        @client.destroy if @client
        stop!
        @config.logger.info('Split SDK shutdown complete')
      end
    end
  end
  @api_key = api_key
  store_flag_sets = config_hash.key?(:flag_sets_filter) ? config_hash[:flag_sets_filter] : []
  store_flag_sets = [] if !store_flag_sets.is_a?(Array)
  flag_sets_count = 0
  flag_sets_invalid = 0
  @config = SplitConfig.new(config_hash.merge(localhost_mode: @api_key == LOCALHOST_API_KEY ))
  if config_hash.key?(:flag_sets_filter)
    flag_sets_count = store_flag_sets.length()
    flag_sets_invalid = flag_sets_count - @config.flag_sets_filter.length()
  end
  raise 'Invalid SDK mode' unless valid_mode
  validate_api_key
  register_factory
  build_telemetry_components
  build_flag_sets_filter
  build_repositories
  build_telemetry_synchronizer(flag_sets_count, flag_sets_invalid)
  build_impressions_sender_adapter
  build_unique_keys_tracker
  build_impressions_components
  @status_manager = Engine::StatusManager.new(@config)
  @split_validator = SplitIoClient::Validators.new(@config)
  @evaluator = Engine::Parser::Evaluator.new(@segments_repository, @splits_repository, @rule_based_segment_repository, @config)
  start!
  fallback_treatment_calculator = SplitIoClient::Engine::FallbackTreatmentCalculator.new(@config.fallback_treatments_configuration) 
  @client = SplitClient.new(@api_key, repositories, @status_manager, @config, @impressions_manager, @evaluation_producer, @evaluator, @split_validator, fallback_treatment_calculator)
  @manager = SplitManager.new(@splits_repository, @status_manager, @config)
end

def register_factory

def register_factory
  SplitIoClient.load_factory_registry
  number_of_factories = SplitIoClient.split_factory_registry.number_of_factories_for(@api_key)
  if(number_of_factories > 0)
    @config.logger.warn("Factory instantiation: You already have #{number_of_factories} factories with this API Key. #{SINGLETON_WARN}")
  elsif(SplitIoClient.split_factory_registry.other_factories)
    @config.logger.warn('Factory instantiation: You already have an instance of the Split factory.' \
      " Make sure you definitely want this additional instance. #{SINGLETON_WARN}")
  end
  SplitIoClient.split_factory_registry.add(@api_key)
end

def repositories

def repositories
  {
    splits: @splits_repository,
    segments: @segments_repository,
    impressions: @impressions_repository,
    events: @events_repository,
    rule_based_segments: @rule_based_segment_repository
  }
end

def start!

def start!
  return start_localhost_components if @config.localhost_mode
  if @config.consumer?
    build_synchronizer
    build_sync_manager
    @sync_manager.start_consumer
    return
  end
  build_fetchers
  build_synchronizer
  build_streaming_components
  build_sync_manager
  @sync_manager.start
end

def start_localhost_components

def start_localhost_components
  LocalhostSplitStore.new(@splits_repository, @config, @status_manager).call
  # Starts thread which loops constantly and cleans up repositories to avoid memory issues in localhost mode
  LocalhostRepoCleaner.new(@impressions_repository, @events_repository, @config).call
end

def stop!

def stop!
  @config.threads.each { |_, t| t.exit }
end

def valid_mode

def valid_mode
  valid_startup_mode = false
  case @config.mode
  when :consumer
    if @config.cache_adapter.is_a? SplitIoClient::Cache::Adapters::RedisAdapter
      if !@config.localhost_mode
        valid_startup_mode = true
      else
        @config.logger.error('Localhost mode cannot be used with Redis. ' \
          'Use standalone mode and Memory adapter instead.')
      end
    else
      @config.logger.error('Consumer mode cannot be used with Memory adapter. ' \
        'Use Redis adapter instead.')
    end
  when :standalone
    if @config.cache_adapter.is_a? SplitIoClient::Cache::Adapters::MemoryAdapter
      valid_startup_mode = true
    else
      @config.logger.error('Standalone mode cannot be used with Redis adapter. ' \
        'Use Memory adapter instead.')
    end
  when :producer
    @config.logger.error('Producer mode is no longer supported. Use Split Synchronizer. ' \
      'See: https://github.com/splitio/split-synchronizer')
  else
    @config.logger.error('Invalid SDK mode selected. ' \
      "Valid modes are 'standalone with memory adapter' and 'consumer with redis adapter'")
  end
  valid_startup_mode
end

def validate_api_key

def validate_api_key
  if(@api_key.nil?)
    @config.logger.error('Factory Instantiation: you passed a nil api_key, api_key must be a non-empty String')
    @config.valid_mode =  false
  elsif (@api_key.empty?)
    @config.logger.error('Factory Instantiation: you passed and empty api_key, api_key must be a non-empty String')
    @config.valid_mode =  false
  end
end