-
Notifications
You must be signed in to change notification settings - Fork 143
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
RFC FS-1148 - Ease conversion between Units of Measure (UoM) and undecorated numerals and simplify casting #784
base: main
Are you sure you want to change the base?
Conversation
- `ResizeArray` | ||
- `Seq` | ||
|
||
Total added methods is 195 `13 supported primitives * 3 methods * (primitive + 4 collection types)`. They are presented as 15 methods with 13 overloads each. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is my personal concern - it's a lot of new surface area and probably a bunch of sigdata, optdata and il.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Incremental size is around 31KB which translates to roughly 163 bytes per method. I don't really have a have a sense of whether that is acceptable or not. If the majority of F# users aren't making extensive use of UoM then there's no sense in making them carry around any extra weight.
The part of the feature that I sense would answer the most questions for users is removal of units from primitives. If I can call LanguagePrimitives.FloatWithMeasure
, I'd expect the inverse LanguagePrimitives.FloatWithoutMeasure
, although the same effect is currently available with the existing primitive conversion methods like float
.
Maybe the collections are better off in a FSharp.Collections.Units
or as an addition to FSharp.UMX
that has an essentially identical API using overloaded methods for primitives extended to non-numeric types?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@T-Gro I'd be happy to accept this PR for an RFC - could you review and if you approve merge? thanks
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe the collections are better off in a FSharp.Collections.Units or as an addition to FSharp.UMX that has an essentially identical API using overloaded methods for primitives extended to non-numeric types?
I agree, move collections from this RFC and keep it tight. Value of collection functions is much smaller as a whole than the primative functions, even in the limit as the number of collections tends to infinity.
In general, currently, I believe that in this form (method per type per action, etc) it should live in a separate library (be part of UMX)? However, I generally think that this particular functionality (add, strip measure) can and should be solved generically (i.e. we should have one type directed generic function |
Based on Don's comments
I took "complex" to mean something I wouldn't be capable of as a novice contributor. Do you have a sense of how complex and pervasive such an addition would be? |
namespace Microsoft.FSharp.Core | ||
|
||
|
||
type Units = |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The purpose of units of measure (including in F#) is to add safety when doing arithmetic and interfacing between systems which may have different conventions. Sometimes measures need to be removed or added when interfacing with systems that do not understand measures. This is inevitable and these are always potential points of failure. To preserve safety here we must have [<RequiresExplicitTypeArguments>]
on all of these. It is then explicit in the code what measure is being added or removed. Without this, the wrong measure (e.g. cm
instead of m
) may get added or removed by mistake, leading to bugs (e.g. numbers wrong by a factor of 100).
What about something like the following (constraints omitted for clarity): [<RequireQualifiedAccess>]
module Measure =
/// Tags a value with a unit of measure.
val inline tag : value:'T -> '``T<'Measure>``
/// Removes a unit of measure from a value.
val inline untag : value:'``T<'Measure>`` -> 'T
/// Tags a value with a new unit of measure.
val inline retag : value:'``T<'Measure1>`` -> '``T<'Measure2>``
/// Maps a value with a unit of measure to another value with a unit of measure.
val inline map : mapping:('T -> 'U) -> value:'``T<'Measure1>`` -> '``U<'Measure2>`` // Tag.
let _ : int<m> = Measure.tag 1
// Untag.
let _ : int = Measure.untag 1<m>
// Retag.
let _ : int<foot> = Measure.retag 1<m>
// Map.
let _ : float<m> = 1<m> |> Measure.map float (Source) It would be trivial to extend to collections if we wanted (at the expense of more overloads). It might be possible to plumb through explicit type parameters if we wanted as well, as suggested in #784 (comment) (maybe someone like @gusty would know how feasible that is). Advantages
(Although in general I would agree with @vzarytovskii's comment #784 (comment) that it would be nicer if this could be in the compiler somehow. Maybe I will think about it.) |
100% agree with @brianrourkeboll that The only downside I see to this approach is on the tooling side examining the function you'll see So you don't know what types support UoM without first providing an incorrect value and looking at the compiler error (which is quite clear to be fair)
Apologies for letting this PR slip under my radar. I got busy then forgot to get back to it. I'd be happy to have another shot at the changes or look into making |
I updated my gist so that the error message in that scenario looks a bit better (it will now show But yes, if we could smooth this out a bit in the compiler instead, perhaps similarly to the other functions that rely on "implicitly augmented" members, maybe that would be nicer. |
static member inline Remove<[<Measure>]'Measure>(input: float<'Measure>):float = retype input | ||
... | ||
static member inline Cast<[<Measure>]'MeasureIn, [<Measure>]'MeasureOut>(input: byte<'MeasureIn>):byte<'MeasureOut> = retype input | ||
static member inline Cast<[<Measure>]'MeasureIn, [<Measure>]'MeasureOut>(input: float<'MeasureIn>):float<'MeasureOut> = retype input |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You can use "1" as input or output unit. Doing this lets you divide the surface area by 3.
open FSharp.Data.UnitSystems.SI.UnitSymbols
let convert<[<Measure>] 'a, [<Measure>] 'b> (a: float<'a>) =
(# "" a: float<'b> #)
1. |> convert<1, m> |> convert<m, kg> |> convert<kg, 1> |> printfn "%g"
Click “Files changed” → “⋯” → “View file” for the rendered RFC.