Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow coercing elgible variants to string/int/float #6311

Merged
merged 7 commits into from
Jun 27, 2023
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@

We've found a bug for you!
/.../fixtures/variant_coercion_float.res:5:10-19

3 │ let x = One(true)
4 │
5 │ let y = (x :> float)
6 │

Type x is not a subtype of float
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@

We've found a bug for you!
/.../fixtures/variant_coercion_float_as.res:5:10-19

3 │ let x = One
4 │
5 │ let y = (x :> float)
6 │

Type x is not a subtype of float
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@

We've found a bug for you!
/.../fixtures/variant_coercion_int.res:5:10-17

3 │ let x = One(true)
4 │
5 │ let y = (x :> int)
6 │

Type x is not a subtype of int
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@

We've found a bug for you!
/.../fixtures/variant_coercion_int_as.res:5:10-17

3 │ let x = One
4 │
5 │ let y = (x :> int)
6 │

Type x is not a subtype of int
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@

We've found a bug for you!
/.../fixtures/variant_coercion_string.res:5:10-20

3 │ let x = One(true)
4 │
5 │ let y = (x :> string)
6 │

Type x is not a subtype of string
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@

We've found a bug for you!
/.../fixtures/variant_coercion_string_as.res:5:10-20

3 │ let x = One
4 │
5 │ let y = (x :> string)
6 │

Type x is not a subtype of string
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
type x = | @as(1.1) One(bool) | @as(2.2) Two

let x = One(true)

let y = (x :> float)
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
type x = | @as(1.1) One | Two

let x = One

let y = (x :> float)
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
type x = | @as(1) One(bool) | @as(2) Two

let x = One(true)

let y = (x :> int)
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
type x = | @as(1) One | Two

let x = One

let y = (x :> int)
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
type x = One(bool) | Two

let x = One(true)

let y = (x :> string)
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
type x = One | @as(2) Two

let x = One

let y = (x :> string)
24 changes: 24 additions & 0 deletions jscomp/ml/ctype.ml
Original file line number Diff line number Diff line change
Expand Up @@ -3951,6 +3951,30 @@ let rec subtype_rec env trace t1 t2 cstrs =
end
| (Tconstr(p1, _, _), _) when generic_private_abbrev env p1 ->
subtype_rec env trace (expand_abbrev_opt env t1) t2 cstrs
| (Tconstr(_, [], _), Tconstr(p, [], _)) when Path.same p Predef.path_string -> (* type coercion for variants represented by strings *)
(match extract_concrete_typedecl env t1 with
| (_, _, {type_kind=Type_variant (constructors)}) ->
if Variant_coercion.can_coerce_to_string constructors then
cstrs
else
(trace, t1, t2, !univar_pairs)::cstrs
| _ -> (trace, t1, t2, !univar_pairs)::cstrs)
| (Tconstr(_, [], _), Tconstr(p, [], _)) when Path.same p Predef.path_int -> (* type coercion for variants represented by ints *)
(match extract_concrete_typedecl env t1 with
| (_, _, {type_kind=Type_variant (constructors)}) ->
if Variant_coercion.can_coerce_to_int constructors then
cstrs
else
(trace, t1, t2, !univar_pairs)::cstrs
| _ -> (trace, t1, t2, !univar_pairs)::cstrs)
| (Tconstr(_, [], _), Tconstr(p, [], _)) when Path.same p Predef.path_float -> (* type coercion for variants represented by floats *)
(match extract_concrete_typedecl env t1 with
| (_, _, {type_kind=Type_variant (constructors)}) ->
if Variant_coercion.can_coerce_to_float constructors then
cstrs
else
(trace, t1, t2, !univar_pairs)::cstrs
| _ -> (trace, t1, t2, !univar_pairs)::cstrs)
| (Tconstr(_, [], _), Tconstr(_, [], _)) -> (* type coercion for records *)
(match extract_concrete_typedecl env t1, extract_concrete_typedecl env t2 with
| (_, _, {type_kind=Type_record (fields1, repr1)}), (_, _, {type_kind=Type_record (fields2, repr2)}) ->
Expand Down
38 changes: 38 additions & 0 deletions jscomp/ml/variant_coercion.ml
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@

type variant_representation = String | Int | Float | Other

let find_as_attribute_payload (attributes : Parsetree.attribute list) =
attributes
|> List.find_map (fun (({txt = name}, payload) : Parsetree.attribute) ->
match (name, payload) with
| ( "as",
PStr
[
{
pstr_desc =
Pstr_eval ({pexp_desc = Pexp_constant constant}, _);
};
] ) ->
Some constant
| _ -> None)

let constructors_representations
(constructors : Types.constructor_declaration list) =
constructors
|> List.map (fun (c : Types.constructor_declaration) ->
match (c.cd_args, c.cd_attributes |> find_as_attribute_payload) with
| Cstr_tuple [], (Some (Pconst_string _) | None) -> String
| Cstr_tuple [], Some (Pconst_integer _) -> Int
| Cstr_tuple [], Some (Pconst_float _) -> Float
| _ -> Other)

(* TODO: Improve error messages? Say why we can't coerce. *)

let can_coerce_to_string (constructors : Types.constructor_declaration list) =
constructors |> constructors_representations |> List.for_all (( = ) String)

let can_coerce_to_int (constructors : Types.constructor_declaration list) =
constructors |> constructors_representations |> List.for_all (( = ) Int)

let can_coerce_to_float (constructors : Types.constructor_declaration list) =
constructors |> constructors_representations |> List.for_all (( = ) Float)
17 changes: 17 additions & 0 deletions jscomp/test/VariantCoercion.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

17 changes: 17 additions & 0 deletions jscomp/test/VariantCoercion.res
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
type a = One | @as("two") Two | Three

let a: a = Three

let b = (a :> string)

type onlyInts = | @as(1) One1 | @as(2) Two2 | @as(3) Three3

let i = One1

let d = (i :> int)

type onlyFloats = | @as(1.1) Onef | @as(2.2) Twof | @as(3.3) Threef

let i = Onef

let d = (i :> float)
3 changes: 2 additions & 1 deletion jscomp/test/build.ninja

Large diffs are not rendered by default.