Sha256: 42c6ea4af365ab9bd8e5a2ad9f6e8acac6a65c4fbc53e7721128ed15d0d8dfb0

Contents?: true

Size: 1.93 KB

Versions: 15

Compression:

Stored size: 1.93 KB

Contents

# frozen_string_literal: true
# typed: true

module T::Private::Abstract::Declare
  Abstract = T::Private::Abstract
  AbstractUtils = T::AbstractUtils

  def self.declare_abstract(mod, type:)
    if AbstractUtils.abstract_module?(mod)
      raise "#{mod} is already declared as abstract"
    end
    if T::Private::Final.final_module?(mod)
      raise "#{mod} was already declared as final and cannot be declared as abstract"
    end

    Abstract::Data.set(mod, :can_have_abstract_methods, true)
    Abstract::Data.set(mod.singleton_class, :can_have_abstract_methods, true)
    Abstract::Data.set(mod, :abstract_type, type)

    mod.extend(Abstract::Hooks)
    mod.extend(T::InterfaceWrapper::Helpers)

    if mod.is_a?(Class)
      if type == :interface
        # Since `interface!` is just `abstract!` with some extra validation, we could technically
        # allow this, but it's unclear there are good use cases, and it might be confusing.
        raise "Classes can't be interfaces. Use `abstract!` instead of `interface!`."
      end

      if Object.instance_method(:method).bind_call(mod, :new).owner == mod
        raise "You must call `abstract!` *before* defining a `new` method"
      end

      # Don't need to silence warnings via without_ruby_warnings when calling
      # define_method because of the guard above

      mod.send(:define_singleton_method, :new) do |*args, &blk|
        result = super(*args, &blk)
        if result.instance_of?(mod)
          raise "#{mod} is declared as abstract; it cannot be instantiated"
        end
        result
      end

      # Ruby doesn not emit "method redefined" warnings for aliased methods
      # (more robust than undef_method that would create a small window in which the method doesn't exist)
      mod.singleton_class.send(:alias_method, :new, :new)

      if mod.singleton_class.respond_to?(:ruby2_keywords, true)
        mod.singleton_class.send(:ruby2_keywords, :new)
      end
    end
  end
end

Version data entries

15 entries across 15 versions & 1 rubygems

Version Path
sorbet-runtime-0.5.11193 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11190 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11188 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11181 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11180 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11178 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11176 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11175 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11170 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11164 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11163 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11162 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11158 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11156 lib/types/private/abstract/declare.rb
sorbet-runtime-0.5.11155 lib/types/private/abstract/declare.rb