gooddata/gooddata-ruby

View on GitHub
lib/gooddata/lcm/actions/synchronize_cas.rb

Summary

Maintainability
B
5 hrs
Test Coverage
# encoding: UTF-8
#
# Copyright (c) 2010-2017 GoodData Corporation. All rights reserved.
# This source code is licensed under the BSD-style license found in the
# LICENSE file in the root directory of this source tree.

require_relative 'base_action'

using TrueExtensions
using FalseExtensions
using IntegerExtensions
using StringExtensions
using NilExtensions

module GoodData
  module LCM2
    class SynchronizeComputedAttributes < BaseAction
      DESCRIPTION = 'Synchronize Computed Attributes'

      PARAMS = define_params(self) do
        description 'Client Used for Connecting to GD'
        param :gdc_gd_client, instance_of(Type::GdClientType), required: true

        description 'Synchronization Info'
        param :synchronize, array_of(instance_of(Type::SynchronizationInfoType)), required: true, generated: true

        description 'Specifies whether to transfer computed attributes'
        param :include_computed_attributes, instance_of(Type::BooleanType), required: false, default: true

        description 'Logger'
        param :gdc_logger, instance_of(Type::GdLogger), required: true

        description 'Additional Hidden Parameters'
        param :additional_hidden_params, instance_of(Type::HashType), required: false

        description 'Abort on error'
        param :abort_on_error, instance_of(Type::StringType), required: false

        description 'Collect synced status'
        param :collect_synced_status, instance_of(Type::BooleanType), required: false

        description 'Sync failed list'
        param :sync_failed_list, instance_of(Type::HashType), required: false
      end

      class << self
        def call(params)
          # set default value for include_computed_attributes
          # (we won't have to do this after TMA-690)
          include_ca = params.include_computed_attributes
          include_ca = true if include_ca.nil?
          include_ca = include_ca.to_b

          results = []
          return results unless include_ca

          client = params.gdc_gd_client
          collect_synced_status = collect_synced_status(params)
          failed_projects = ThreadSafe::Array.new

          params.synchronize.each do |info|
            from = info.from
            to_projects = info.to

            params.gdc_logger.info "Synchronize Computed Attributes from project pid: #{from}"

            to_projects.peach do |entry|
              ca_scripts = entry[:ca_scripts]
              next unless ca_scripts

              pid = entry[:pid]
              next if sync_failed_project(pid, params)

              ca_chunks = ca_scripts[:maqlDdlChunks]
              to_project = client.projects(pid)
              unless to_project
                process_failed_project(pid, "Invalid 'to' project specified - '#{pid}'", failed_projects, collect_synced_status)
                next
              end

              params.gdc_logger.info "Synchronizing Computed Attributes to project: '#{to_project.title}', PID: #{pid}"
              error_message = nil
              begin
                ca_chunks.each { |chunk| to_project.execute_maql(chunk) }
              rescue => e
                error_message = "Error occurred when executing MAQL for project: \"#{to_project.title}\" reason: \"#{e.message}\", chunks: #{ca_chunks.inspect}"
                process_failed_project(pid, error_message, failed_projects, collect_synced_status)
              end

              results << {
                from: from,
                to: pid,
                status: error_message.nil? ? 'ok' : 'failed'
              }
            end
          end

          process_failed_projects(failed_projects, short_name, params) if collect_synced_status
          results
        end
      end
    end
  end
end