openjaf/cenit

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

Summary

Maintainability
B
5 hrs
Test Coverage
require 'xsd/include_missing_exception'

module Setup
  class Schema < Validator
    include SnippetCode
    include Setup::FormatValidator

    legacy_code_attribute :schema

    build_in_data_type.with(:namespace, :uri, :snippet).referenced_by(:namespace, :uri)

    # TODO shared_deny :simple_generate, :bulk_generate

    field :uri, type: String
    field :schema_type, type: StringifiedSymbol

    belongs_to :schema_data_type, class_name: Setup::JsonDataType.to_s, inverse_of: nil

    attr_readonly :uri

    validates_presence_of :uri, :schema
    validates_uniqueness_of :uri, scope: :namespace

    def code_extension
      case schema_type
      when :xml_schema
        '.xsd'
      when :json_schema
        '.json'
      else
        super
      end
    end

    def schema
      code
    end

    def schema=(sch)
      self.code = sch
    end

    def title
      uri
    end

    def validates_before
      super
      self.name = uri unless name.present?
      self.schema = schema.strip
      self.schema_type =
        if schema.start_with?('{') || schema.start_with?('[')
          :json_schema
        else
          :xml_schema
        end
    end

    def data_format
      schema_type.to_s.split('_').first.to_sym
    end

    def validate_data(data)
      case schema_type
      when :json_schema
        unless data.is_a?(Hash)
          data =
            if data.respond_to?(:to_hash)
              data.to_hash
            elsif data.respond_to?(:to_json)
              JSON.parse(data.to_json)
            else
              JSON.parse(data.to_s)
            end
        end
        Mongoff::Validator.validate_instance(
          data,
          schema: JSON.parse(schema),
          data_type: self.class.data_type
        )
        []
      when :xml_schema
        unless data.is_a?(Nokogiri::XML::Document)
          data = (data.respond_to?(:to_xml) ? data.to_xml : data).to_s unless data.is_a?(String)
          data = Nokogiri::XML(data)
        end
        Nokogiri::XML::Schema(cenit_ref_schema).validate(data)
      end
    rescue Exception => ex
      [ex.message]
    end

    def find_ref_schema(ref)
      if ref == uri
        self
      else
        ns = namespace
        if ref.is_a?(Hash)
          ns = ref['namespace']
          ref = ref['name']
        end
        (sch = Setup::Schema.where(namespace: ns, uri: ref).first) &&
          sch.schema
      end
    end

    def parse_schema
      @parsed_schema ||=
        case schema_type
        when :json_schema
          parse_json_schema
        when :xml_schema
          parse_xml_schema
        else
          # TODO: !!!
        end
    end

    def json_schemas
      bind_includes
      if parse_schema.is_a?(Hash)
        @data_type_name = @parsed_schema.keys.first
        @parsed_schema
      else
        json_schms = @parsed_schema.json_schemas
        if (elements_schms = json_schms.keys.select { |name| name.start_with?('element') }).size == 1
          @data_type_name = elements_schms.first
        end
        json_schms
      end
    end

    def bind_includes
      return if @includes_binded
      parse_schema.bind_includes(namespace_ns) unless parse_schema.is_a?(Hash)
      @includes_binded = true
    end

    def included?(qualified_name, visited = Set.new)
      return false if visited.include?(self) || visited.include?(@parsed_schema)
      visited << self
      if parse_schema.is_a?(Hash)
        @parsed_schema.key?(qualified_name)
      else
        @parsed_schema.included?(qualified_name, visited)
      end
    end

    def cenit_ref_schema(options = {})
      options = {
        service_url: Cenit.routed_service_url,
        schema_service_path: Cenit.schema_service_path
      }.merge(options)
      send("cenit_ref_#{schema_type}", options)
    end

    private

    def cenit_ref_json_schema(_options = {})
      schema
    end

    def cenit_ref_xml_schema(options = {})
      doc = Nokogiri::XML(schema)
      cursor = doc.root.first_element_child
      while cursor
        if %w(import include redefine).include?(cursor.name) && (attr = cursor.attributes['schemaLocation'])
          token = Cenit::TenantToken.create data: { ns: namespace, uri: abs_uri(uri, attr.value) },
                                            token_span: 1.hour
          attr.value = "#{options[:service_url]}#{"/#{options[:schema_service_path]}".squeeze('/')}?token=#{token.token}"
        end
        cursor = cursor.next_element
      end
      doc.to_xml
    end

    def parse_json_schema
      { uri => JSON.parse(schema) }
    end

    def parse_xml_schema
      Xsd::Document.new(uri, schema).schema
    end

    def abs_uri(base_uri, uri)
      self.class.abs_uri(base_uri, uri)
    end

    class << self

      def abs_uri(base_uri, uri)
        uri = URI.parse(uri.to_s)
        return uri.to_s unless uri.relative?

        base_uri = URI.parse(base_uri.to_s)
        uri = uri.to_s.split('/')
        path = base_uri.path.split('/')
        begin
          path.pop
        end while uri[0] == '..' ? uri.shift && true : false

        path = (path + uri).join('/')

        uri = URI.parse(path)
        uri.scheme = base_uri.scheme
        uri.host = base_uri.host
        uri.to_s
      end

    end

  end
end