Declaring types for non-methods
Sorbet provides the most value when it has a wealth of programmer-supplied static types.
However, because Sorbet implements a gradual type system, it treats most definitions without explicit annotations as untyped. This means that Sorbet can only use static types for methods, constants, instance variables, and class variables if they are accompanied with explicit static types.
For more information on why type annotations are required in Sorbet, see Why does Sorbet sometimes need type annotations?.
Type annotations for methods are provided using a sig
before the method
definition. Method type annotations are
described in great detail on the Method Signatures.
Other type annotations are provided using the T.let
type assertion.
Annotating local variables
Sorbet does not usually need type annotations for local variables, as it can
infer the type of the local variable based on how it is initialized. For
example, in the following program, Sorbet can tell x
is an Integer
based on
the fact that it is initialized with an expression that evaluates to an
Integer
:
x = 2 + 3
You may still provide a wider type annotation if you would like. This can occasionally be helpful if you want the type of a variable to be broader than Sorbet’s inferred type, such as in situations where you are changing the value of a variable in a loop or block to something that is broader than the expression used to initialize the variable:
# without this T.let, x would have the inferred type NilClass
x = T.let(nil, T.nilable(Integer))
(0..10).each do |n|
x = n if n % 3 == 0
end
Annotating constants
Sorbet does very minimal inference for types of constants. These are the cases where Sorbet infers constant types:
Constants initialized with simple literals (like
"foo"
,123
, or[""]
) will have their types inferred. Importantly, this does not includeHash
literals.Constants initialized as an alias to another constant, like
X = Y
. Sorbet infers the type ofX
to be whatever the type ofY
is (ifY
has neither an explicit nor an inferred type,X
will be inferred to beT.untyped
).Array literal type inference is recursive, so Sorbet will only assume an array type for an array literal if the contents of the array are all literals (including class or module constant literals).
Array literals are inferred to have type
T::Array[...]
where...
is a union type of all the types of elements in the array. If the array literal is also frozen with the.freeze
method, the inferred type will be a tuple type instead. This array vs tuple decision is not recursive, because.freeze
is not recursive. To have Sorbet infer array of tuples types, call.freeze
on each array nested inside the top-level array.Constants initialized with a call to
SomeClass.new
will have their type inferred toSomeClass
. Importantly, this assumption happens regardless of whether thenew
method actually returns an instance ofSomeClass
, which might not be the case if thenew
method has been overridden.In these cases, Sorbet reports an error stating that it requires an explicit type annotation to correct the faulty assumption. This only applies at the top-level (e.g., not inside arrays).
In all other cases, Sorbet does not infer the types of constants, and will
assume a type of T.untyped
. In # typed: strict
files, Sorbet
reports an error requiring that a type be specified, so that Sorbet does not
assume T.untyped
.
To specify the type of a constant, use T.let
:
NAMES = T.let(["Nelson", "Dmitry", "Paul"], T::Array[String])
In codebases that require calling .freeze
on constants, the call to .freeze
must go inside the T.let
, or Sorbet will not see the call to T.let
.
# ✅ Good
NAMES = T.let(["Nelson", "Dmitry", "Paul"].freeze, T::Array[String])
# ^^^^^^^ ✅
# ❌ BAD
NAMES = T.let(["Nelson", "Dmitry", "Paul"], T::Array[String]).freeze
# ❌ ^^^^^^^
We recommend using the rubocop-sorbet
gem, which modifies the behavior of
RuboCop’s Style/MutableConstant
rule, making it aware of T.let
.
Declaring class and instance variables
To declare the static type of an instance variable, we can use T.let
in a
class’s constructor:
class MyObj
def initialize
@foo = T.let(0, Integer)
end
end
We can also declare class variables and instance variables on a singleton class
using T.let
at the top-level of a class:
class HasVariables
# Class variable
@@llamas = T.let([], T::Array[Llama])
# Instance variable on the singleton class
@alpaca_count = T.let(0, Integer)
end
Sorbet requires that instance and class variables are defined in these specific places to guarantee that they’re initialized. But sometimes requiring that these variables be declared in specific places is too restrictive. Sorbet allows an instance variable to be declared anywhere so long as the type is at least nilable:
class A
def foo
# Does NOT have to be declared in `initialize`, because it's nilable:
@x = T.let(0, T.nilable(Integer))
end
def self.bar
# Also works for `self.` methods:
@y = T.let('', T.nilable(String))
end
end
Declaring lazily-initialized instance variables
Sorbet also supports T.let
type annotations for instance variables that are
lazily initialized with ||=
, just like those initialized eagerly with =
. The
syntax looks just the same:
module B
sig {returns(String)}
def current_user
@user ||= T.let(ENV.fetch('USER'), T.nilable(String))
end
end
Note that the same restrictions about the variable being declared T.nilable
apply, but that Sorbet’s control flow-sensitive typing is
smart enough to understand that either:
@user
has already been initialized to a non-nil value, so the||
condition is truthy and thus must return aString
, or@user
has not yet been initialized, but the initial value, computed usingENV.fetch('USER')
, has typeString
(and is thus non-nil).
Note that using ||=
like this only works when nil
is the same as
“uninitialized.” If it’s possible for the instance variable to be initialized
and also possibly nil
(meaning that there’s no need to attempt to
re-initialize it on subsequent calls), use the defined?
keyword built into
Ruby:
module B
sig {returns(T.nilable(String))}
def current_git_dir
return @git_dir if defined?(@git_dir)
@git_dir = T.let(ENV['GIT_DIR'], T.nilable(String))
end
end
Limitations on instance variable inference
A current shortcoming of Sorbet is that in many cases it cannot reuse static
type knowledge in order to automatically determine the type of an instance or
class variable. In the following example, Sorbet will naturally understand that
@x
is of type Integer
, but it cannot determine the static type of @y
without a T.let
and therefore treats it as T.untyped
when used in other
methods:
class Foo
sig {params(x: Integer, y: Integer).void}
def initialize(x, y)
@x = x
@y = y + 0
end
sig {void}
def example
T.reveal_type(@x) # Integer
T.reveal_type(@y) # T.untyped
end
end
Sorbet can only infer the types of instance variables in a relatively specific
context: in particular, only when that instance variable is initialized to the
exact value of a parameter to the constructor in the body of the constructor. In
other cases, you will need to use T.let
to explicitly give the types of
instance variables.
Note: This particular limitation is because of how Sorbet performs typechecking: it needs to know the types of instance variables before it typechecks method definitions, but even a simple expression like
y + 0
will require a typechecking pass to determine what its resulting type is, which means Sorbet won’t be able to tell what type to infer for an instance variable until after it has already started typechecking the method where instance variables are defined. Sorbet resolves this cyclical dependency by either using types that you have explicitly defined withT.let
, or by relying on simple code patterns like the one described above.