id | title | sidebar_label |
---|---|---|
sigs |
Method Signatures |
Signatures |
This page describes the syntax of method signatures, or
sig
s. For a complete reference of the types available for use within asig
, see the "Type System" section to the left.
Method signatures are the primary way that we enable static and dynamic type checking in our code. In this document, we'll answer:
- How to add signatures to methods.
- Why we'd want to add signatures in the first place.
Signatures are valid Ruby syntax. To be able to write signatures, we first
extend T::Sig
at the top of our class or module:
extend T::Sig
The basic syntax looks like this:
sig {params(x: SomeType, y: SomeOtherType).returns(MyReturnType)}
def foo(x, y); ...; end
It's also possible to break a sig
up across multiple lines. Here's the same
signature as above, rearranged:
sig do
params(
x: SomeType,
y: SomeOtherType,
)
.returns(MyReturnType)
end
def foo(x, y); ...; end
In every signature, there is an optional params
section, and a required
returns
section. Here's a complete example:
# typed: true
require 'sorbet-runtime'
class Main
# Bring the `sig` method into scope
extend T::Sig
sig {params(x: String).returns(Integer)}
def self.main(x)
x.length
end
end
In the sig
we refer to all parameters by their name, regardless of whether
it's a positional, keyword, block, or rest parameter. Once we've annotated the
method, Sorbet will automatically infer the types of any local variables we use
in the method body.
Here's the syntax for required and optional positional parameters:
sig do
params(
x: String, # required positional param
y: String, # optional positional param
z: T.nilable(String) # optional *AND* nilable param
)
.returns(String)
end
def self.main(x, y = 'foo', z = nil)
x + y + (z ? z : '')
end
Here's the syntax for required and optional keyword parameters:
sig do
params(
x: String, # required keyword param
y: String, # optional keyword param
z: T.nilable(String) # optional *AND* nilable keyword param
)
.void
end
def self.main(x:, y: 'foo', z: nil)
# ...
end
Sometimes called splats. There are two kinds of rest parameters: "all the
arguments" (*args
) and "all the keyword arguments" (**kwargs
):
Types for rest parameters frequently trip people up. There's a difference between what's written in the sig annotation and what type that variable has in the method body:
sig do
params(
# Integer describes a single element of args
args: Integer, # rest positional params
# Float describes a single value of kwargs
kwargs: Float # rest keyword params
)
.void
end
def self.main(*args, **kwargs)
# Positional rest args become an Array in the method body:
T.reveal_type(args) # => Revealed type: `T::Array[Integer]`
# Keyword rest args become a Hash in the method body:
T.reveal_type(kwargs) # => Revealed: type `T::Hash[Symbol, Float]`
end
Notice that in the sig, args
is declared as Integer
, but in the method body
Sorbet knows that args
is actually a T::Array[Integer]
because it can see
from the method definition that args
is a rest parameter.
It's similar for kwargs
: it's declared as Float
, but in the method body
Sorbet knows that it'll be a Hash from Symbol
keys to Float
values.
Note: The choice to use this syntax for annotating rest parameters in Sorbet was informed by precedent in other languages (most notably Scala).
sig do
params(
blk: T.proc.returns(NilClass)
)
.void
end
def self.main(&blk)
# ...
end
See Blocks, Procs and Lambda Types for more information on how to write type annotations for a method's block parameter.
When a method has no parameters, omit the params
from the sig
:
sig {returns(Integer)}
def self.main
42
end
See the next section for more information.
Unlike params
, we have to tell Sorbet what our method returns, even if it
has "no useful return." For example, consider this method:
def main
5.times do
puts 'Hello, world!'
end
end
We care more about what effect this method has (printing to the screen) than
what this method returns (5
). We could write a sig
like this:
sig {returns(Integer)} # ← Problematic! Read why below...
This is annoying for a bunch of reasons:
-
We'd get a useless type error if someone added
puts 'Goodbye, world!'
at the bottom ofmain
. Instead of returning5
(Integer
), the method would now returnnil
(NilClass
). -
Call sites in untyped code can implicitly depend on us always returning an
Integer
. For example, what if people think returning5
is actually some sort of exit code?
Instead, Sorbet has a special way to mark methods where we only care about the
effect: void
:
sig {void}
Using void
instead of returns(...)
does a number of things:
-
Statically,
srb
will let us return any value (for example, returning either5
ornil
is valid). -
Also statically,
srb
will error when typed code tries to inspect the result of avoid
method. -
In the runtime,
sorbet-runtime
will throw away the result of our method, and return a dummy value instead. (Allvoid
methods return the same dummy value.) This prevents untyped code from silently depending on what we return.
Concretely, here's a full example of how to use void
to type methods with
useless returns:
# typed: true
require 'sorbet-runtime'
class Main
extend T::Sig
# (1) greet has a useless return:
sig {params(name: String).void}
def self.greet(name)
puts "Hello, #{name}!"
end
# (2) name_length must be given a string:
sig {params(name: String).returns(Integer)}
def self.name_length(name)
name.length
end
end
# (3) It's an error to pass a void result to name_length:
Main.name_length(Main.greet('Alice')) # => error!
There are many ways to define class (static) methods in Ruby. How a method is
defined changes where the extend T::Sig
line needs to go. These are the two
preferred ways to define class methods with sigs:
-
def self.greet
class Main # In this style, at the top level of the class extend T::Sig sig {params(name: String).void} def self.greet(name) puts "Hello, #{name}!" end end
-
class << self
class Main class << self # In this style, inside the `class << self` extend T::Sig sig {params(name: String).void} def greet(name) # ... end end end
Taking a step back, why do we need sig
s in the first place?
Sorbet does type inference for local variables within methods, and then requires annotations for method parameters and return types. This mix of type inference and type annotations balances being explicit with being powerful:
- With a small amount of information, Sorbet can power autocompletion results and catch type errors.
- Since there's no type inference across methods, each method can be typechecked 100% in parallel, for fast performance. Other people can't write code which makes typechecking your code slow.
- Method signatures serve as machine-checked documentation for whoever reads the code.
So basically: the complexity of Ruby requires it, it enables Sorbet to be performant, and it encourages better development practices. Anecdotally, we've seen all three of these things have a positive effect on development.
For example, Sorbet could have re-used YARD annotations, or extended Ruby with new syntax.
There are a number of reasons why we have type annotations as valid Ruby method calls:
-
The existing ecosystem of Ruby tooling still works.
Editor syntax highlighting, Ruby parsers, RuboCop, IDEs, and text editors, and more all work out of the box with Sorbet's type annotations.
-
No runtime changes required.
If Sorbet introduced new syntax, type-annotated code would no longer be directly runnable simply with
ruby
at the command line. This means no build step is required, and no special changes to the core language. -
Runtime checking is a feature.
In a gradual type system like Sorbet, the static checks can be turned off at any time. Having runtime-validated type annotations gives greater confidence in the predictions that
srb
makes statically. -
Type assertions in code would be inevitable.
Having constructs like
T.let
andT.cast
work in line requires that type annotations already be syntactically valid Ruby (havingT.let
andT.cast
to do type refinements and assertions are central to Sorbet being a gradual type system). Since types must already be valid Ruby, it makes sense to havesig
s be valid Ruby too.