Elixir now performs inference of whole functions. The best way to show the new capabilities are with examples. Take the following code:
def add_foo_and_bar(data) do
data.foo + data.bar
end
Elixir now infers that the function expects a map
as first argument, and the map must have the keys .foo
and .bar
which values of either integer()
or float()
. The return type will be either integer()
or float()
.
Here is another example:
def sum_to_string(a, b) do
Integer.to_string(a + b)
end
Even though the +
operator works with both integers and floats, Elixir infers that a
and b
must be both integers, as the result of +
is given to a function that expects an integer. The inferred type information is then used during type checking to find possible typing errors.
This release also adds type checking when dispatching and implementing protocols.
For example, string interpolation in Elixir uses the String.Chars
protocol. If you pass a value that does not implement said protocol, Elixir will now emit a warning accordingly.
Here is an example passing a range, which cannot be converted into a string, to an interpolation:
defmodule Example do
def my_code(first..last//step = range) do
"hello #{range}"
end
end
the above emits the following warnings:
warning: incompatible value given to string interpolation:
data
it has type:
%Range{first: term(), last: term(), step: term()}
but expected a type that implements the String.Chars protocol, it must be one of:
dynamic(
%Date{} or %DateTime{} or %NaiveDateTime{} or %Time{} or %URI{} or %Version{} or
%Version.Requirement{}
) or atom() or binary() or float() or integer() or list(term())
Warnings are also emitted if you pass a data type that does not implement the Enumerable
protocol as a generator to for-comprehensions:
defmodule Example do
def my_code(%Date{} = date) do
for(x <- date, do: x)
end
end
will emit:
warning: incompatible value given to for-comprehension:
x <- date
it has type:
%Date{year: term(), month: term(), day: term(), calendar: term()}
but expected a type that implements the Enumerable protocol, it must be one of:
dynamic(
%Date.Range{} or %File.Stream{} or %GenEvent.Stream{} or %HashDict{} or %HashSet{} or
%IO.Stream{} or %MapSet{} or %Range{} or %Stream{}
) or fun() or list(term()) or non_struct_map()
https://elixir-lang.org/blog/2025/02/26/elixir-openchain-certification/
- [Access] Add
Access.values/0
for traversing maps and keyword lists values - [Calendar] Support 2-arity options for
Calendar.strftime/3
which receives the whole data type - [Code] Add
:migrate_call_parens_on_pipe
formatter option - [Code.Fragment] Preserve more block content around cursor in
container_cursor_to_quoted
:migrate_call_parens_on_pipe
formatter option - [Enum] Provide more information on
Enum.OutOfBoundsError
- [Kernel] Allow controlling which applications are used during inference
- [Kernel] Support
min/2
andmax/2
as guards - [Macro] Print debugging results from
Macro.dbg/1
as they happen, instead of once at the end - [Protocol] Type checking of protocols dispatch and implementations
- [ExUnit.CaptureLog] Parallelize log dispatch when multiple processes are capturing log
- [ExUnit.Doctest] Support ellipsis in doctest exceptions to match the remaining of the exception
- [ExUnit.Doctest] Add
:inspect_opts
option for doctest
- [IEx.Autocomplete] Functions annotated with
@doc group: "Name"
metadata will appear within their own groups in autocompletion
- [mix] Add support for
MIX_PROFILE_FLAGS
to configureMIX_PROFILE
- [mix compile] Add
Mix.Tasks.Compile.reenable/1
- [mix deps.compile] Support
MIX_OS_DEPS_COMPILE_PARTITION_COUNT
for compiling deps concurrently across multiple operating system processes - [mix help] Add
mix help Mod
,mix help :mod
,mix help Mod.fun
andmix help Mod.fun/arity
- [mix xref graph] Add support for
--format json
- [mix xref graph] Emit a warning if
--source
is part of a cycle
- [DateTime] Do not truncate microseconds regardless of precision in
DateTime.diff/3
- [File] Properly handle permissions errors cascading from parent in
File.mkdir_p/1
- [Stream] Ensure
Stream.transform/1
respects suspend command when its inner stream halts - [URI] Several fixes to
URI.merge/2
related to trailing slashes, trailing dots, and hostless base URIs
- [mix cmd] Preserve argument quoting in subcommands
- [mix xref graph] Provide more consistent output by considering strong connected components only when computing graphs
- [Node]
Node.start/2-3
is deprecated in favor ofNode.start/2
with a keyword list
- [mix compile]
--no-protocol-consolidation
is deprecated in favor of--no-consolidate-protocols
for consistency withmix.exs
configuration - [mix compile.protocols] Protocol consolidation is now part of
compile.elixir
and has no effect
- [Code] The
on_undefined_variable: :warn
is deprecated. Relying on undefined variables becoming function calls will not be supported in the future - [File] Passing a callback as third argument to
File.cp/3
is deprecated, pass it as aon_conflict: callback
option instead - [File] Passing a callback as third argument to
File.cp_r/3
is deprecated, pass it as aon_conflict: callback
option instead - [Kernel] The struct update syntax, such as
%URI{uri | path: "/foo/bar"}
is deprecated in favor of pattern matching on the struct when the variable is defined and then using the map update syntax%{uri | path: "/foo/bar"}
. Thanks to the type system, pattern matching on structs can find more errors, more reliably - [Kernel.ParallelCompiler] Passing
return_diagnostics: true
as an option is required oncompile
,compile_to_path
andrequire
- [Logger] The
:backends
configuration is deprecated, either set the:default_handler
to false or start backends in your application start callback
- [mix] The
:default_task
,:preferred_cli_env
, and:preferred_cli_target
configuration insidedef project
in yourmix.exs
has been deprecated in favor of:default_task
,:preferred_envs
and:preferred_targets
inside thedef cli
function - [mix do] Using commas as task separator in
mix do
(such asmix do foo, bar
) is deprecated, use+
instead (as inmix do foo + bar
)
The CHANGELOG for v1.18 releases can be found in the v1.18 branch.