[erlang-questions] Inconsistent and incomprehensible handling of variables in spec's by compiler
Kostis Sagonas
kostis@REDACTED
Fri Feb 17 21:56:00 CET 2012
On 02/17/2012 10:54 AM, Robert Virding wrote:
> If in my file I add the sepc:
>
> -spec set_var(PrefixExp, Val, State) -> State.
>
> I get the compiler errors, errors mind you, not warnings:
>
> src/luerl_eval.erl:332: type variable 'PrefixExp' is only used once (is unbound)
> src/luerl_eval.erl:332: type variable 'Val' is only used once (is unbound)
>
> but if I write it as:
>
> -spec set_var(_PrefixExp, _Val, State) -> State.
>
> then the compiler is happy and is quiet. If I write it as:
>
> -spec set_var(Val, Val, State) -> State.
>
> (which is wrongly from my point of view) or go all verbose and write:
>
> -spec set_var(PrefixExp::any(), Val::any(), State::any()) -> State::any().
>
> (which doesn't really add any useful information) the compiler is again happy and silent.
>
> This is inconsistent and does not make sense for variables in a function spec. Either it is wrong, for some incomprehensible reason, to have singleton variables in a function spec, or it shouldn't matter what the variable names are. And why should adding a type, even the most general one, make a difference to the variable name usage. And why should the compiler be bothered with the function spec at all seeing it is doesn't use it and it is not part of the language.
>
> In my naive view of the the typing world my original spec says that set_var/3 takes three arguments of any type and returns a value of the same type as the third argument. Then why not just add type information and make the compiler happy? Well in my case it doesn't give me anything; it makes the spec much longer and there is nothing in a type which would give me more information than is in the variable names. Also dialyzer doesn't actually *need* the spec as it works it out by itself anyway. And the type of PrefixExp is a big hairy recursive type which I am happy to let dialyzer work out for me. Typer gives me a half-screen long type for it.
>
> I get the same behaviour when running dialyzer on the different versions. Again I don't understand why it behaves like that.
As others mentioned, in their most basic form, specs should contain
*types*, not type variables. Continuing your example, the following is a
valid spec:
-spec set_var(exp(), val(), state()) -> state().
for some appropriate definitions of exp(), val() and state() types.
I recommend that you write such specs. I would even go as far as say
that you should stick to this kind of specs: specs that contain types
and types alone.
Of course, the benefits of writing the above spec are dependent on the
precision of your type declarations. If you are lazy and map all three
types to any(), something that the type language allows you to do, then
of course you cannot expect much from the tools that take these specs
into account, can you? (In such a case the spec is there mostly (only?)
for documentation purposes. Personally, I am not interested in this type
of specs; they have very little purpose.)
Now, for convenience and for compatibility with what the edoc @spec
language also allowed, we have retained the possibility that the user
provides, perhaps selectively, *names* to argument positions and return
results. So, you can write the above as:
-spec set_var(PrefixExp::exp(), ImpValue::val(), state()) -> state().
Note that PrefixExp and ImpValue, i.e. whatever comes to the left of the
:: symbol in the above, are *optional names* for arguments 1 and 2, not
type variables.
Type variables in specs come into the picture when you want to express
(subtype or equality) relations between different argument positions or
return values. (And by definition to express a relation you need to
somehow use the same variable twice; personally I cannot think of a case
where a singleton type variable is interesting.) For example, you may
want to state in a more emphatic way that the third argument and the
return result of the set_var/3 function are of the same type. You can do
that as follows:
-spec set_var(PrefixExp::exp(), ImpValue::val(), InState::ST) ->
OutState::ST when ST::state().
(Note that the only type variable in the above is ST.)
Another example of use of type variables is in functions like map/2.
Since the type system is based on subtyping, you would need to write its
spec as follows:
-spec map(F::fun((X) -> Y), InL::[XE]) -> OutL::[YE]
when X::term(), Y::term(), XE::term(), YE::term(), XL::X, YL::Y.
or, even stricter, as:
-spec map(F::fun((X) -> Y), InL::[X]) -> OutL::[Y]
when X::term(), Y::term().
but I guess most people would find the above two ways of giving a spec
for map/2 probably too verbose so we also allowed them to write simply:
-spec map(fun((X) -> Y), [X]) -> [Y].
with the implicit assumption that for all type variables T that are not
qualified by a subtype constraint, there is a constraint T::term() which
is implicit.
(Note that in this case, all type variables are not singletons.)
Also, because we had a hunch that there may be Erlang programmers who
have hard time grasping the subtleties of the type language (for
example, they might be confusing type variables with argument and
position names) and might make typos in specs, we decided to never allow
type variables that are singleton. As I explained, such variables are
not just unconstrained, but they are really not needed because they
serve no purpose; instead type names do.
I saw/see no reason for one to write a type variable just to name a
position; if you think otherwise, please provide an example that makes
sense. Till you do, you are not allowed to write MY_NAME for the type
any(); if you want the type any() please write any(); if you want to
name it appropriately, you are required to write MY_NAME::any() or its
shorthand MY_NAME::_.
I hope it is clear now.
Kostis
More information about the erlang-questions
mailing list