openjaf/cenit

View on GitHub
app/models/setup/base_pull.rb

Summary

Maintainability
F
6 days
Test Coverage
module Setup
  class BasePull < Setup::Task
    include UploaderHelper

    abstract_class

    build_in_data_type

    mount_uploader :pull_request, AccountUploader
    mount_uploader :pulled_request, AccountUploader

    def pull_request_hash
      @pull_request_hash ||= hashify(pull_request)
    end

    def pulled_request_hash
      @pulled_request_hash ||= hashify(pulled_request)
    end

    def run(message)
      clear_hashes
      if pull_request_hash.present?
        pull_request_hash[:install] = message[:install].to_b if ask_for_install?
        if (pull_parameters = message[:pull_parameters]).is_a?(Hash)
          pull_request_hash[:pull_parameters].merge!(pull_parameters)
          source_shared_collection.parametrize(pull_request_hash[:collection_data] || {}, pull_request_hash[:pull_parameters], keep_values: true)
        end
        pulled_request_hash = self.class.pull(source_shared_collection, pull_request_hash)
        self.remove_pull_request = true
        {
          fixed_errors: :warning,
          errors: :error
        }.each do |key, type|
          (pulled_request_hash[key] || []).each do |msg|
            notify(message: msg, type: type)
          end
        end
        if (missing_parameters = pulled_request_hash[:missing_parameters]).present?
          notify(message: "Missing parameters (IDs: #{missing_parameters.to_sentence})")
        end
        store pulled_request_hash.to_json, on: pulled_request
      else
        self.remove_pulled_request = true
        msg_pull_request = message.dup
        msg_pull_request[:updated_records_ids] = true
        pull_request_hash = self.class.pull_request(source_shared_collection, msg_pull_request)
        store pull_request_hash.to_json, on: pull_request
        if message[:skip_pull_review].to_b
          notify(message: 'Skipping pull review', type: :warning)
          run(message)
        else
          self.progress = [progress, 45].max
          notify(message: 'Waiting for pull review', type: :warning)
          resume_manually
        end
      end
    end

    protected

    def clear_hashes
      @pull_request_hash = @pulled_request_hash = nil
    end

    def source_shared_collection
      fail NotImplementedError
    end

    def ask_for_install?
      false
    end

    def hashify(uploader)
      JSON.parse(uploader.read || '{}').with_indifferent_access
    rescue Exception => ex
      raise "Invalid JSON #{uploader.mounted_as}: #{ex.message}"
    end

    class << self

      def pull_request(shared_collection, options = {})
        pull_parameters = options[:pull_parameters] || {}
        missing_parameters = []
        unless options[:ignore_missing_parameters]
          shared_collection.each_pull_parameter do |pull_parameter|
            param_id = pull_parameter.id.to_s
            if pull_parameter.required? && !pull_parameters.key?(param_id)
              missing_parameters << param_id
            end
          end
        end

        new_records = Hash.new { |h, k| h[k] = [] }
        updated_records = Hash.new { |h, k| h[k] = [] }

        pull_data = shared_collection.data_with(pull_parameters)

        invariant_data = {}

        collection_data = { '_reset' => resetting = [] }
        collection = Setup::Collection.where(name: shared_collection.name).first
        fields = %w(readme title)
        unless collection && fields.all? { |field| Cenit::Utility.eql_content?(collection.send(field), shared_collection.send(field)) }
          fields.each do |field|
            shared_value = shared_collection[field]
            unless collection && Cenit::Utility.eql_content?(collection[field], shared_value)
              collection_data[field] = shared_value
              resetting << field
            end
          end
        end
        fields = %w(metadata)
        unless collection && fields.all? { |field| Cenit::Utility.eql_content?(collection.send(field), pull_data[field]) }
          fields.each do |field|
            shared_value = pull_data[field]
            unless collection && Cenit::Utility.eql_content?(collection[field], shared_value)
              collection_data[field] = shared_value
              resetting << field
            end
          end
        end

        Setup::Collection.reflect_on_all_associations(:has_and_belongs_to_many).each do |relation|
          entry = relation.name.to_s
          if (items = pull_data[entry])
            invariant_data[entry] = invariant_names = Set.new
            invariant_on_collection = 0
            items_data_type = relation.klass.data_type
            refs =
              items.collect do |item|
                criteria = ref_criteria = {}
                items_data_type.get_referenced_by.each do |field|
                  field = field.to_s
                  if %w(id _id).include?(field)
                    criteria['_id'] = item['_id'] || item['id']
                  else
                    criteria[field] = item[field]
                  end
                end
                criteria.delete_if { |_, value| value.nil? }
                criteria = Cenit::Utility.deep_remove(criteria, '_reference')
                unless (on_collection = (record = collection && Cenit::Utility.find_record(criteria, collection.send(relation.name))))
                  record = Cenit::Utility.find_record(criteria, relation.klass)
                end
                if record
                  share_hash_options = {}
                  if shared_collection.installed?
                    share_hash_options[:include_id] = ->(r) { r.is_a?(Setup::CrossOriginShared) && r.shared? }
                  end
                  record_hash = record.share_hash(share_hash_options)
                  record_hash = Cenit::Utility.stringfy(record_hash)
                  if item['_type']
                    record_hash['_type'] = record.class.to_s unless record_hash['_type']
                  end
                  record_hash['_reset'] ||= item['_reset'] if item['_reset']
                  record_hash.reject! { |key, _| !item.has_key?(key) }
                  invariant = Cenit::Utility.eql_content?(record_hash, item) do |record_value, item_value, hash_key|
                    if %w(id _id _primary).include?(hash_key)
                      record_value.nil? || item_value.nil?
                    else
                      (record_value.nil? && item_value.blank?) || (item_value.nil? && record_value.blank?)
                    end
                  end
                  if invariant
                    invariant_names << ref_criteria
                    invariant_on_collection += 1 if on_collection
                    item = {}
                  else
                    item.delete_if { |key, _| ref_criteria.key?(key) }
                    updated_records[entry] <<
                      if options[:updated_records_ids]
                        record.id.to_s
                      else
                        record
                      end
                  end
                  item['id'] = record.id.to_s
                  check_embedded_items(item, record)
                else
                  new_records[entry] << item
                end
                item
              end
            unless (collection && collection.send(relation.name).count == invariant_on_collection) && (invariant_on_collection == items.size)
              collection_data[entry] = refs
              resetting << entry
            end
          elsif collection && collection.send(relation.name).present?
            resetting << entry
          end
        end

        if resetting.present? && !options[:discard_collection]
          if collection
            updated_records['collections'] <<
              if options[:updated_records_ids]
                collection.id.to_s
              else
                collection
              end
          else
            new_records['collections'] << { 'name' => shared_collection.name }
          end
        end

        invariant_data.each do |key, invariant_names|
          pull_data[key].delete_if do |item|
            criteria = { namespace: item['namespace'], name: item['name'] }
            criteria.delete_if { |_, value| value.nil? }
            invariant_names.include?(criteria)
          end
        end

        [collection_data, pull_data, updated_records].each { |hash| hash.each_key { |key| hash.delete(key) if hash[key].blank? } }

        {
          pull_parameters: pull_parameters,
          new_records: new_records,
          updated_records: updated_records,
          missing_parameters: missing_parameters,
          pull_data: pull_data,
          collection_data: collection_data,
          collection_discarded: options[:discard_collection].present?,
          updated_records_ids: options[:updated_records_ids].present?
        }
      end

      def pull(shared_collection, pull_request = {})
        pull_request = pull_request.with_indifferent_access
        pull_request = pull_request(shared_collection, pull_request) if pull_request[:pull_data].nil?
        errors = []
        if pull_request[:missing_parameters].blank?
          created_nss_ids = []
          begin
            collection_data = pull_request.delete(:collection_data)
            (collection_data['namespaces'] || []).each do |ns_hash|
              if (slug = ns_hash['slug']) && Setup::Namespace.where(slug: slug).present?
                ns_hash.delete('slug')
              end
              next if ns_hash.has_key?('id')
              if (ns = Setup::Namespace.create_from_json(ns_hash)).errors.blank?
                ns_hash['id'] = ns.id.to_s
                created_nss_ids << ns.id
              else
                fail "Error creating namespace for #{ns_hash.to_json}: #{ns.errors.full_messages.to_sentence}"
              end
            end
            collection = Setup::Collection.where(name: shared_collection.name).first
            attrs = (collection && collection.attributes.deep_dup) || {}
            attrs.delete('_id')
            collection = Setup::Collection.new(attrs)
            collection.from_json(collection_data, add_only: true, skip_refs_binding: true)
            collection.events.each { |e| e[:activated] = false if e.is_a?(Setup::Scheduler) && e.new_record? }
            begin
              collection.name = BSON::ObjectId.new.to_s
            end while Setup::Collection.where(name: collection.name).present?
            unless Cenit::Utility.save(collection,
                                       bind_references: { if: ->(r) { r.instance_variable_get(:@_edi_parsed) } },
                                       create_collector: (create_collector = Set.new),
                                       saved_collector: (saved = Set.new))

              collection.errors.full_messages.each { |msg| errors << msg }
              collection.errors.clear
              if Cenit::Utility.save(collection, { create_collector: create_collector })
                pull_request[:fixed_errors] = errors.collect { |error| "Auto-fixed error: #{error}" }
                errors = []
              else
                saved.each do |obj|
                  if (obj = (obj.reload rescue nil))
                    obj.delete
                  end
                end
              end
            end
            if errors.blank?
              unless pull_request[:collection_discarded]
                Setup::Collection.where(name: shared_collection.name).delete
                collection.name = shared_collection.name
                collection.image = shared_collection.image if shared_collection.image.present?
                collection.save(add_dependencies: false)
                shared_collection.pulled(collection: collection,
                                         install: pull_request[:install])
                pull_request[:collection] = { id: collection.id.to_s }
              end
              pull_data = pull_request.delete(:pull_data)
              pull_request[:created_records] = collection.inspect_json(inspecting: :id, inspect_scope: create_collector).reject { |_, value| !value.is_a?(Enumerable) }
              collection.destroy if pull_request[:collection_discarded]
              pull_request[:pull_data] = pull_data
            end
          rescue Exception => ex
            n = Setup::SystemReport.create_from(ex, 'Pulling ERROR')
            errors << "An unexpected error occurs (#{ex.message}). Ask for support by supplying this code: #{n.id}"
          end
          unless errors.blank?
            Setup::Namespace.all.any_in(id: created_nss_ids.to_a).delete_all
            pull_request[:errors] = errors
          end
        end
        pull_request
      end

      private

      def check_embedded_items(item, record)
        (item_model = record.class).model_properties_schemas.each do |property, schema|
          next if schema['referenced']
          next unless (property_value = item[property]) && (property_model = item_model.property_model(property))
          next unless (property_data_type = property_model.data_type).get_referenced_by.present?
          if schema['type'] == 'object'
            if (record_value = record.send(property))
              criteria = {}
              property_data_type.get_referenced_by.each { |field| criteria[field.to_s] = property_value[field.to_s] }
              criteria.delete_if { |_, value| value.nil? }
              if Cenit::Utility.match?(record_value, criteria)
                property_value['id'] = record_value.id.to_s
                check_embedded_items(property_value, record_value)
              end
            end
          else
            if (reset = item['_reset'])
              reset = [reset] unless reset.is_a?(Array)
            else
              reset = []
            end
            reset << property
            item['_reset'] = reset
            if (association = record.send(property).to_a).present?
              property_value.each do |sub_item|
                criteria = {}
                property_data_type.get_referenced_by.each { |field| criteria[field.to_s] = sub_item[field.to_s] }
                criteria.delete_if { |_, value| value.nil? }
                if (sub_record = Cenit::Utility.find_record(criteria, association))
                  sub_item['id'] = sub_record.id.to_s
                  check_embedded_items(sub_item, sub_record)
                end
              end
            end
          end
        end
      end
    end

  end
end