08-16-2023 01:49 PM
Hi,
I am in a custom business rule.
I have a calculate formula like the following one, which works fine:
api.Data.Calculate("A#Sales1 = A#Sales2 + 50”)
Now let say I want to declare a variable instead of having the 50 typed in the formula string, and pass that to the Calculate, what is the correct syntax to do that? Is it possible?
I would like to do something like this (the one below is not the correct syntax, but it gives an idea of what I am trying to do):
Dim Growth As Decimal = 50
api.Data.Calculate("A#Sales1 = A#Sales2 + Growth”)
Thank you
Solved! Go to Solution.
08-16-2023 07:28 PM - edited 08-16-2023 07:31 PM
The safest approach is this:
Api.Data.FormulaVariables.SetDecimalVariable("Growth", 50)
API.data.Calculate("A#Sales1 = A#Sales2 + $Growth”)
This approach avoids possible issues with converting numbers to string - when the user executing it has a different Culture, or when decimals get truncated (hence losing precision). There is a blog post discussing it as part of internationalising an application.
08-16-2023 01:52 PM
Just a note to say that in the actual formula I am doing the additional dimensional filtering in the Calculate, the formula above is only for exemplification
08-16-2023 01:58 PM - edited 08-16-2023 02:00 PM
Here you go:
Dim Growth As Decimal = 50 api.Data.Calculate($"A#Sales1 = A#Sales2 + {Growth}”)
08-17-2023 05:58 AM
This formula gives me the following error:
"Unable to execute Business Rule 'Plan'. Invalid calculation script. Unexpected text. {Growth}."
08-31-2023 11:57 AM
The method Robb mentions is using string interpolation. The error message you sent to me sounds like you are missing the dollar sign ($) before the literal string that tells the complier it is an interpolated string.
08-16-2023 07:28 PM - edited 08-16-2023 07:31 PM
The safest approach is this:
Api.Data.FormulaVariables.SetDecimalVariable("Growth", 50)
API.data.Calculate("A#Sales1 = A#Sales2 + $Growth”)
This approach avoids possible issues with converting numbers to string - when the user executing it has a different Culture, or when decimals get truncated (hence losing precision). There is a blog post discussing it as part of internationalising an application.
08-31-2023 10:12 PM
Hey Jack, just wanted to understand how this issue occurs a little more.
Are you saying
- BRs get executed in the OneStream culture setting of the user
- So when Growth gets converted to a string as in Robb's example then it will convert a decimal to a string in the format of the user, so if user is French and Growth is 50.123/50,123 it will result in something like API.data.Calculate("A#Sales1 = A#Sales2 + 50,123”)? Which wouldn't work because OneStream would be expecting API.data.Calculate("A#Sales1 = A#Sales2 + 50.123”)?
09-01-2023 04:13 AM
Formulas expect numbers formatted in Invariant Culture (basically US), but in some contexts the process is actually executed under the user's own Culture, resulting in the behaviour you describe.
In addition, even if you convert from number to string in Invariant, there is a significant chance that numbers will be truncated, losing precision.
So the best approach is to never convert Decimals to String at all, and to assign them to a Formula Variable instead.
09-01-2023 10:24 AM
I thought there were a few methods that does this in culture invariant way, the XFToString ones?
09-01-2023 10:35 AM
There are, but they might lose precision on long numbers, as digits could be truncated (or so I'm told).
09-01-2023 10:38 AM
My understanding with .net is that string interpolation of a decimal value preserves precision unless explicity told not to. I think @JackLacava makes an interesting point about culture, due to disparate use of decimal points and commas. I'm not convinced that effects the precision of a calculation as much as the presentation of the results?
Dim value As Decimal = 123.456789D
Dim formattedString As String = $"The value is {value}" ' This will maintain the precision of the decimal
Dim value As Decimal = 123.456789D
Dim formattedString As String = $"The value is {value:F2}" ' This will format the value to 2 decimal places
Dim value As Decimal = 123456.789D
Dim frenchCulture As New System.Globalization.CultureInfo("fr-FR")
Dim frenchFormattedString As String = String.Format(frenchCulture, "The value in French format is {0:N}", value)
' The output will be: "The value in French format is 123 456,789"
Dim value As Decimal = 123456.789D
Dim japaneseCulture As New System.Globalization.CultureInfo("ja-JP")
Dim japaneseFormattedString As String = String.Format(japaneseCulture, "The value in Japanese format is {0:N}", value)
' The output will be: "The value in Japanese format is 123,456.789"
09-01-2023 10:54 AM - edited 09-01-2023 11:04 AM
No precision lost in this test comparing parsed interpolated strings to straight Decimals:
09-01-2023 11:01 AM - edited 09-01-2023 11:12 AM
WriteLine converts to String though, doesn't it...? So if there is truncation, you're just showing it being done twice.
But good for trying things out, I'd be happy to be disproven. Generally speaking, internally we've been told to push formula variables as best practice, since it's definitely the safest option.
09-01-2023 11:27 AM - edited 09-04-2023 11:29 AM
This is a good discussion. it highlights some important things in computer science in general.
@JackLacava Your question "WriteLine converts to String though, doesn't it...? So if there is truncation, you're just showing it being done twice." got me thinking...
The following code demonstrates the parsing of an interpolated decimal over a million accumulating iterations of a calculation and compares it to the same million iterations of direct decimal calculation. I think this qualifies interpolation as not changing precision.
** The key to this test is the comparison of the results is made by the code, before being written to the screen as String values.