View Source Changelog for Elixir v1.17
This release includes type inference of patterns to provide warnings for an initial set of constructs (binaries, maps, and atoms) within the same function. It also includes a new Duration data type to interact with Calendar types, support for Erlang/OTP 27, and many other improvements.
Warnings from gradual set-theoretic types
This release introduces gradual set-theoretic types to infer types from patterns and use them to type check programs, enabling the Elixir compiler to find faults and bugs in codebases without requiring changes to existing software. The underlying principles, theory, and roadmap of our work have been outlined in "The Design Principles of the Elixir Type System" by Giuseppe Castagna, Guillaume Duboc, José Valim.
At the moment, Elixir developers will interact with set-theoretic types only through warnings found by the type system. The current implementation models all data types in the language:
binary(),integer(),float(),pid(),port(),reference()- these types are indivisible. This means both1and13get the sameinteger()type.atom()- it represents all atoms and it is divisible. For instance, the atom:fooand:hello_worldare also valid (distinct) types.map()and structs - maps can be "closed" or "open". Closed maps only allow the specified keys, such as%{key: atom(), value: integer()}. Open maps support any other keys in addition to the ones listed and their definition starts with..., such as%{..., key: atom(), value: integer()}. Structs are closed maps with the__struct__key.tuple(),list(), andfunction()- currently they are modelled as indivisible types. The next Elixir versions will also introduce fine-grained support to them.
We focused on atoms and maps on this initial release as they are respectively the simplest and the most complex types representations, so we can stress the performance of the type system and quality of error messages. Modelling these types will also provide the most immediate benefits to Elixir developers. Assuming there is a variable named user, holding a %User{} struct with an address field, Elixir v1.17 will emit the following warnings at compile-time:
Pattern matching against a map or a struct that does not have the given key, such as
%{adress: ...} = user(noticeaddressvsadress)Accessing a key on a map or a struct that does not have the given key, such as
user.adressInvoking a function on non-modules, such as
user.address()Capturing a function on non-modules, such as
&user.address/0Attempting to invoke to call an anonymous function without an actual function, such as
user.()Performing structural comparisons with structs, such as
my_date < ~D[2010-04-17]Performing structural comparisons between non-overlapping types, such as
integer >= stringBuilding and pattern matching on binaries without the relevant specifiers, such as
<<name>>(this warns because by default it expects an integer, it should have been<<name::binary>>instead)Attempting to rescue an undefined exception or a struct that is not an exception
Accessing a field that is not defined in a rescued exception
These new warnings help Elixir developers find bugs earlier and give more confidence when refactoring code, especially around maps and structs. While some of these warnings were emitted in the past, they were discovered using syntax analysis. The new warnings are more reliable, precise, and with better error messages. Keep in mind, however, that the Elixir typechecker only infers types from patterns within the same function at the moment. Analysis from guards and across function boundaries will be added in future relases. For more details, see our new reference document on gradual set-theoretic types.
The type system was made possible thanks to a partnership between CNRS and Remote. The development work is currently sponsored by Fresha, Starfish*, and Dashbit.
Erlang/OTP support
This release adds support for Erlang/OTP 27 and drops support for Erlang/OTP 24. We recommend Elixir developers to migrate to Erlang/OTP 26 or later, especially on Windows. Support for WERL (a graphical user interface for the Erlang terminal on Windows) will be removed in Elixir v1.18.
Adding Duration and shift/2 functions
Elixir introduces the Duration data type and APIs to shift dates, times, and date times by a given duration, considering different calendars and time zones.
iex> Date.shift(~D[2016-01-31], month: 2)
~D[2016-03-31]Note the operation is called shift (instead of add) since working with durations does not obey properties such as associativity. For instance, adding one month and then one month does not give the same result as adding two months:
iex> ~D[2016-01-31] |> Date.shift(month: 1) |> Date.shift(month: 1)
~D[2016-03-29]Still, durations are essential for building intervals, recurring events, and modelling scheduling complexities found in the world around us. For DateTimes, Elixir will correctly deal with time zone changes (such as Daylight Saving Time), but provisions are also available in case you want to surface conflicts (for example, you shifted to a wall clock that does not exist, because the clock has been moved forward by one hour). See DateTime.shift/2 for examples.
Finally, a new Kernel.to_timeout/1 function has been added, which helps developers normalize durations and integers to a timeout used by Process APIs. For example, to send a message after one hour, one can now write:
Process.send_after(pid, :wake_up, to_timeout(hour: 1))v1.17.1 (2024-06-18)
1. Enhancements
Mix
- [mix compile.elixir] Do not run fixpoint computation on runtime dependencies. This should considerably improve compilation times for large projects when changing only one or two files
2. Bug fixes
EEx
- [EEx] Do not warn for assignment with blocks in EEx
Elixir
- [Kernel] Fix bug when using pinned variables inside
with'selsepatterns - [Kernel] Fix Dialyzer error when with else clause is calling a
no_returnfunction
ExUnit
- [ExUnit] Do not alternative sync/async suites on
--repeat-until-failure
v1.17.0 (2024-06-12)
1. Enhancements
Elixir
- [Access] Add
Access.find/1that mirrorsEnum.find/2 - [Code] Support cursor inside fn/rescue/catch/else/after inside
Code.Fragment.container_cursor_to_quoted/2 - [Date] Add
Date.shift/2to shift dates with duration and calendar-specific semantics - [Date] Allow
Dateto accept years outside of-9999..9999range - [DateTime] Add
DateTime.shift/2to shift datetimes with duration and calendar-specific semantics - [Duration] Add a new
Durationdata type - [GenServer] Add
GenServer.format_status/1callback - [Kernel] Add
Kernel.get_in/1with safe nil-handling for access and structs - [Kernel] Add
Kernel.is_non_struct_map/1guard - [Kernel] Add
Kernel.to_timeout/1 - [Kernel] Emit warnings for undefined functions from modules defined within the same context as the caller code
- [Kernel] Support integers in uppercase sigils
- [Keyword] Add
Keyword.intersect/2-3to mirror theMapAPI - [Macro] Add
Macro.Env.define_alias/4,Macro.Env.define_import/4,Macro.Env.define_require/4,Macro.Env.expand_alias/4,Macro.Env.expand_import/5, andMacro.Env.expand_require/6to aid the implementation of language servers and embedded languages - [NaiveDateTime] Add
NaiveDateTime.shift/2to shift naive datetimes with duration and calendar-specific semantics - [Process] Add
Process.set_label/1 - [String] Add
String.byte_slice/3to slice a string to a maximum number of bytes while keeping it UTF-8 encoded - [System] Support
use_stdio: falseinSystem.cmd/3andSystem.shell/2 - [Time] Add
Time.shift/2to shift times with duration and calendar-specific semantics
ExUnit
- [ExUnit] Propagate the test process itself as a caller in
start_supervised - [ExUnit] Include max cases in ExUnit reports
IEx
- [IEx.Helpers] Warns if
recompilewas called and the current working directory changed - [IEx.Helpers] Add
c/0as an alias tocontinue/0 - [IEx.Pry] Add
IEx.Pry.annotate_quoted/3to annotate a quoted expression with pry breakpoints
Logger
- [Logger] Format
:gen_statemreports using Elixir data structures - [Logger] Include process label in logger events
Mix
- [mix deps] Add
:depthoption toMix.SCM.Git, thus supporting shallow clones of Git dependencies - [mix deps] Warn if
:optionalis used in combination with:in_umbrella - [mix deps.get] Do not add optional dependency requirements if its parent dep was skipped
- [mix deps.tree] Add
--umbrella-onlytomix deps.tree - [mix profile.tprof] Add a new profiler, available on Erlang/OTP 27+, which can measure count, time, and heap usage
- [mix test] Add
mix test --breakpointsthat sets up a breakpoint before each test that will run - [mix test] Add
mix test --repeat-until-failureto rerun tests until a failure occurs - [mix test] Add
mix test --slowest-modulesto print slowest modules based on all of the tests they hold - [mix test] Generate cover HTML files in parallel
2. Bug fixes
Elixir
- [bin/elixir.bat] Improve handling of quotes and exclamation marks in flags
- [Code] Address a bug where AST nodes for
(a -> b)were not wrapped as part of the literal encoder - [Kernel] Resolve inconsistencies of how
..and...are handled at the AST level - [Kernel] Fix parsing precedence of ambiguous operators followed by containers
- [Kernel] Do not expand code in
quote bind_quoted: ...twice - [Kernel] Respect
:lineproperty when:fileis given as option toquote - [Kernel] Do not crash on
Macro.escape/2when passing a quote triplet without valid meta - [Kernel] Avoid double tracing events when capturing a function
- [Kernel] Fix a bug where captured arguments would conflict when a capture included a macro that also used captures
- [Module] Return default value in
Module.get_attribute/3for persisted attributes which have not yet been written to
IEx
- [IEx.Helpers] Update the history size whenever history is pruned
Mix
- [mix deps] Fix error message for diverged SCM definition in sibling
3. Soft deprecations (no warnings emitted)
Elixir
- [GenServer] Deprecate
GenServer.format_status/2callback to align with Erlang/OTP 25+
Mix
- [mix profile.cprof] Deprecated in favor of the new
mix profile.tprof - [mix profile.eprof] Deprecated in favor of the new
mix profile.tprof
4. Hard deprecations
Elixir
- [IO] Passing
:alltoIO.read/2andIO.binread/2is deprecated, pass:eofinstead - [Kernel] Single-quote charlists are deprecated, use
~cinstead - [Kernel] Deprecate escaping closing delimiter in uppercase sigils
- [Range]
left..rightwithout explicit steps inside patterns and guards is deprecated, writeleft..right//stepinstead - [Range] Decreasing ranges, such as
10..1without an explicit step is deprecated, write10..1//-1instead
ExUnit
- [ExUnit.Case]
register_test/4is deprecated in favor ofregister_test/6for performance reasons
v1.16
The CHANGELOG for v1.16 releases can be found in the v1.16 branch.