# frozen_string_literal: true module Environments class EnvironmentsByDeploymentsFinder attr_reader :project, :current_user, :params def initialize(project, current_user, params = {}) @project = project @current_user = current_user @params = params end # rubocop: disable CodeReuse/ActiveRecord def execute deployments = project.deployments deployments = if ref deployments_query = params[:with_tags] ? 'ref = :ref OR tag IS TRUE' : 'ref = :ref' deployments.where(deployments_query, ref: ref.to_s) elsif commit deployments.where(sha: commit.sha) else deployments.none end environments = if Feature.enabled?(:environments_by_deployments_finder_exists_optimization, default_enabled: :yaml) project.environments.available .where('EXISTS (?)', deployments.where('environment_id = environments.id')) else environment_ids = deployments .group(:environment_id) .select(:environment_id) project.environments.available .where(id: environment_ids) end if params[:find_latest] find_one(environments.order_by_last_deployed_at_desc) else find_all(environments.order_by_last_deployed_at.to_a) end end # rubocop: enable CodeReuse/ActiveRecord private def find_one(environments) [environments.find { |environment| valid_environment?(environment) }].compact end def find_all(environments) environments.select { |environment| valid_environment?(environment) } end def valid_environment?(environment) # Go in order of cost: SQL calls are cheaper than Gitaly calls return false unless Ability.allowed?(current_user, :read_environment, environment) return false if ref && params[:recently_updated] && !environment.recently_updated_on_branch?(ref) return false if ref && commit && !environment.includes_commit?(commit) true end def ref params[:ref].try(:to_s) end def commit params[:commit] end end end