Forum Discussion

AndreaF's avatar
AndreaF
Contributor III
2 years ago

Using a variable in the calculate string formula

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

 

  • JackLacava's avatar
    JackLacava
    Honored Contributor

    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.

    • DanielWillis's avatar
      DanielWillis
      Valued Contributor

      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”)?

      • JackLacava's avatar
        JackLacava
        Honored Contributor

        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.

  • RobbSalzmann's avatar
    RobbSalzmann
    Valued Contributor II

    No precision lost in this test comparing parsed interpolated strings to straight Decimals:

     

     

    • JackLacava's avatar
      JackLacava
      Honored Contributor

      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.

  • RobbSalzmann's avatar
    RobbSalzmann
    Valued Contributor II

    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.

     

     

     

  • RobbSalzmann's avatar
    RobbSalzmann
    Valued Contributor II

    Here you go:

    Dim Growth As Decimal = 50
    api.Data.Calculate($"A#Sales1 = A#Sales2 + {Growth}”)
    • AndreaF's avatar
      AndreaF
      Contributor III

      This formula gives me the following error:

      "Unable to execute Business Rule 'Plan'. Invalid calculation script. Unexpected text. {Growth}."

      • Big_Rick_CPM's avatar
        Big_Rick_CPM
        Contributor

        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. 

  • RobbSalzmann's avatar
    RobbSalzmann
    Valued Contributor II

    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"
  • AndreaF's avatar
    AndreaF
    Contributor III

    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