Skip to content

Latest commit

 

History

History
91 lines (70 loc) · 1.98 KB

self-type.md

File metadata and controls

91 lines (70 loc) · 1.98 KB
id title
self-type
T.self_type

TODO: This page is still a fragment. Contributions welcome!

Warning: This feature is experimental and has known limitations. It may not work as expected or change without notice.

T.self_type

This type can be used in return types to indicate that calling this method on will return the same type as the type of the receiver (the receiver is the thing we call a method on i.e., x in x.foo). For instance, #dup returns T.self_type. No matter what class you call it on, you will get back the same type.

# typed: true

class Parent
  extend T::Sig

  sig {returns(T.self_type)}
  def foo
    self
  end
end

class Child < Parent; end

T.reveal_type(Parent.new.foo) # Revealed type: Parent
T.reveal_type(Child.new.foo) # Revealed type: Child

module Mixin
  extend T::Sig

  sig {returns(T.self_type)}
  def bar
    self
  end
end

class UsesMixin
  extend Mixin
end

T.reveal_type(UsesMixin.bar) # Revealed type: T.class_of(UsesMixin)

Certain advanced use cases of T.self_type are not supported:

class Generic < Parent
  extend T::Generic
  TM = type_member

  sig {returns(Generic[T.self_type])} # error: Only top-level T.self_type is supported
  def bad
    Generic[T.untyped].new
  end
end

Note that T.self_type declares that the type should be exactly what the receiver of the method is. It is not useful for declaring the type of factory methods. For these types of methods, use T.attached_class instead:

class Parent
  # sig {returns(T.self_type)} # WRONG
  sig {returns(T.attached_class)}
  def self.make
    new
  end
end

class Child < Parent; end

T.reveal_type(Parent.make) # => `Parent`
T.reveal_type(Child.make)  # => `Child`

If the above example was declared using T.self_type, Sorbet would expect to see the make method return a value of type T.class_of(Parent), but new returns a value of type Parent (an instance of the class, not the class object itself).