module Eco module API class Session class Batch class Jobs < API::Common::Session::BaseSession DELAY_BETWEEN_JOBS = 0.3 include Enumerable attr_reader :name def initialize(e, name:) super(e) @name = name reset end def reset @jobs = {} @callbacks = {} end def length count end def empty? count == 0 end def each(&block) return to_enum(:each) unless block items.each(&block) end def items @jobs.values end def [](name) @jobs[name] end def exists?(name) @jobs.key?(name) end # It retrieves an existing job named `name` or creates it if it doesn't exist. # @note # - the block should only be passed when creating the job # @param [see @Eco::API::Session::Batch::Jobs#new] # @return [Eco::API::Session::Batch::Job] def job(name, type: nil, sets: nil, usecase: nil, accept_update_with_no_id: false, &block) fatal "Can't give a job post-launch callback &block to an already existing job" if exists?(name) new(name, type: type, sets: sets, usecase: usecase, accept_update_with_no_id: accept_update_with_no_id, &block) unless exists?(name) self[name] end # Creates a new `Batch::Job` included as part of this `Batch::Jobs`. # @param [see Eco::API::Session::Job#new] # @yield [job, status] callback after launching the batch job request against the server. # @yieldparam job [Eco::API::Session::Batch::Job] the job we have launched against the server. # @yieldparam status [Eco::API::Session::Batch::Status] the status of the batch job launch. # @return [Eco::API::Session::Batch::Job] def new(name, type:, sets:, usecase: nil, accept_update_with_no_id: false, &block) fatal "Can't create job named '#{name}' because it already exists." if exists?(name) Batch::Job.new(enviro, name: name, type: type, sets: sets, usecase: usecase, accept_update_with_no_id: accept_update_with_no_id).tap do |job| add(job, &block) end end # @param job [Eco::API::Session::Batch::Job] the `Batch::Job` to add. # @yield [job, status] callback after launching the batch job request against the server. # @yieldparam job [Eco::API::Session::Batch::Job] the job we have launched against the server. # @yieldparam status [Eco::API::Session::Batch::Status] the status of the batch job launch. # @return [Eco::API::Session::Batch::Job] def add(job, &block) fatal "Expected Eco::API::Session::Batch::Job object. Given #{job.class}" unless job.is_a?(Eco::API::Session::Batch::Job) @jobs[job.name] = job @callbacks[job] = block if block_given? end def pending? any? {|job| job.pending?} end # Launches every `Batch::Job` in the group. # @note # - if there was post-launch callback for a `Batch::Job`, it calls it. # @return [Hash] def launch(simulate: false) each_with_index do |job, idx| if job.pending? status[job] = job_status = job.launch(simulate: simulate) callback = @callbacks[job] callback.call(job, job_status) if callback Eco::API::Session::Batch::JobsGroups.counter(delay_between_jobs) if !simulate && idx < self.length - 1 end end launch(simulate: simulate) if pending? return status end def find_jobs(type:) each_with_object([]) do |job, jbs| jbs.push(job) if job.type == type end end def status if block_given? status.each do |job, job_status| yield(job, job_status) end self else @jobs_status ||= {} end end def errors? any? {|job| job.errors?} end def summary [].tap do |msg| map {|job| msg << job.summary} end.join("\n") end private def delay_between_jobs config.delay_between_jobs || DELAY_BETWEEN_JOBS end end end end end end