Skip to content
New issue

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

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

Already on GitHub? # to your account

Feat/expr in var origin #45

Merged
merged 18 commits into from
Mar 20, 2025
5 changes: 3 additions & 2 deletions Numscript.g4
Original file line number Diff line number Diff line change
Expand Up @@ -21,13 +21,14 @@ valueExpr:
| monetaryLit # monetaryLiteral
| left = valueExpr op = DIV right = valueExpr # infixExpr
| left = valueExpr op = (PLUS | MINUS) right = valueExpr # infixExpr
| LPARENS valueExpr RPARENS # parenthesizedExpr;
| LPARENS valueExpr RPARENS # parenthesizedExpr
| functionCall # application;

functionCallArgs: valueExpr ( COMMA valueExpr)*;
functionCall:
fnName = (OVERDRAFT | IDENTIFIER) LPARENS functionCallArgs? RPARENS;

varOrigin: EQ functionCall;
varOrigin: EQ valueExpr;
varDeclaration:
type_ = IDENTIFIER name = VARIABLE_NAME varOrigin?;
varsDeclaration: VARS LBRACE varDeclaration* RBRACE;
Expand Down
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -32,5 +32,5 @@ require (
github.com/tidwall/match v1.1.1 // indirect
github.com/tidwall/pretty v1.2.1 // indirect
github.com/tidwall/sjson v1.2.5 // indirect
golang.org/x/exp v0.0.0-20240707233637-46b078467d37
golang.org/x/exp v0.0.0-20240707233637-46b078467d37 // indirect
)
21 changes: 13 additions & 8 deletions internal/analysis/check.go
Original file line number Diff line number Diff line change
Expand Up @@ -162,7 +162,7 @@ func (res *CheckResult) check() {
}

if varDecl.Origin != nil {
res.checkVarOrigin(*varDecl.Origin, varDecl)
res.checkExpression(*varDecl.Origin, varDecl.Type.Name)
}
}
}
Expand Down Expand Up @@ -302,18 +302,20 @@ func (res *CheckResult) checkDuplicateVars(variableName parser.Variable, decl pa
}
}

func (res *CheckResult) checkVarOrigin(fnCall parser.FnCall, decl parser.VarDeclaration) {
resolution, ok := Builtins[fnCall.Caller.Name]
if ok {
resolution, ok := resolution.(VarOriginFnCallResolution)
if ok {
res.fnCallResolution[decl.Origin.Caller] = resolution
res.assertHasType(decl.Name, resolution.Return, decl.Type.Name)
func (res *CheckResult) checkFnCall(fnCall parser.FnCall) string {
returnType := TypeAny

if resolution, ok := Builtins[fnCall.Caller.Name]; ok {
if resolution, ok := resolution.(VarOriginFnCallResolution); ok {
res.fnCallResolution[fnCall.Caller] = resolution
returnType = resolution.Return
}
}

// this must come after resolution
res.checkFnCallArity(&fnCall)

return returnType
}

func (res *CheckResult) checkExpression(lit parser.ValueExpr, requiredType string) {
Expand Down Expand Up @@ -379,6 +381,9 @@ func (res *CheckResult) checkTypeOf(lit parser.ValueExpr, typeHint string) strin
case *parser.StringLiteral:
return TypeString

case *parser.FnCall:
return res.checkFnCall(*lit)

default:
return TypeAny
}
Expand Down
4 changes: 3 additions & 1 deletion internal/analysis/hover.go
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,7 @@ func hoverOnVar(varDecl parser.VarDeclaration, position parser.Position) Hover {
}

if varDecl.Origin != nil {
hover := hoverOnFnCall(*varDecl.Origin, position)
hover := hoverOnExpression(*varDecl.Origin, position)
if hover != nil {
return hover
}
Expand Down Expand Up @@ -183,6 +183,8 @@ func hoverOnExpression(lit parser.ValueExpr, position parser.Position) Hover {
return hover
}

case *parser.FnCall:
return hoverOnFnCall(*lit, position)
}

return nil
Expand Down
10 changes: 8 additions & 2 deletions internal/cmd/run.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@
var overdraftFeatureFlag bool
var oneOfFeatureFlag bool
var accountInterpolationFlag bool
var midScriptFunctionCallFeatureFlag bool

type inputOpts struct {
Script string `json:"script"`
Expand Down Expand Up @@ -129,6 +130,9 @@
if accountInterpolationFlag {
featureFlags[interpreter.ExperimentalAccountInterpolationFlag] = struct{}{}
}
if midScriptFunctionCallFeatureFlag {
featureFlags[interpreter.ExperimentalMidScriptFunctionCall] = struct{}{}
}

Check warning on line 135 in internal/cmd/run.go

View check run for this annotation

Codecov / codecov/patch

internal/cmd/run.go#L133-L135

Added lines #L133 - L135 were not covered by tests

result, err := interpreter.RunProgram(context.Background(), parseResult.Value, opt.Variables, interpreter.StaticStore{
Balances: opt.Balances,
Expand Down Expand Up @@ -208,9 +212,11 @@
cmd.Flags().BoolVar(&runStdinFlag, "stdin", false, "Take input from stdin (same format as the --raw option)")

// Feature flag
cmd.Flags().BoolVar(&overdraftFeatureFlag, interpreter.ExperimentalOverdraftFunctionFeatureFlag, false, "feature flag to enable the overdraft() function")
cmd.Flags().BoolVar(&oneOfFeatureFlag, interpreter.ExperimentalOneofFeatureFlag, false, "feature flag to enable the oneof combinator")

cmd.Flags().BoolVar(&overdraftFeatureFlag, interpreter.ExperimentalOverdraftFunctionFeatureFlag, false, "enables the experimental overdraft() function")
cmd.Flags().BoolVar(&oneOfFeatureFlag, interpreter.ExperimentalOneofFeatureFlag, false, "enable the experimental oneof combinator")

Check warning on line 217 in internal/cmd/run.go

View check run for this annotation

Codecov / codecov/patch

internal/cmd/run.go#L215-L217

Added lines #L215 - L217 were not covered by tests
cmd.Flags().BoolVar(&accountInterpolationFlag, interpreter.ExperimentalAccountInterpolationFlag, false, "enables an account interpolation syntax, e.g. @users:$id:pending")
cmd.Flags().BoolVar(&midScriptFunctionCallFeatureFlag, interpreter.ExperimentalMidScriptFunctionCall, false, "allows to use function call as expression, and to use any expression when definining variables")

Check warning on line 219 in internal/cmd/run.go

View check run for this annotation

Codecov / codecov/patch

internal/cmd/run.go#L219

Added line #L219 was not covered by tests

// Output options
cmd.Flags().StringVar(&runOutFormatOpt, "output-format", OutputFormatPretty, "Set the output format. Available options: pretty, json.")
Expand Down
63 changes: 63 additions & 0 deletions internal/interpreter/balances.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
package interpreter

import (
"math/big"

"github.com/formancehq/numscript/internal/utils"
)

func (b Balances) fetchAccountBalances(account string) AccountBalance {
return defaultMapGet(b, account, func() AccountBalance {
return AccountBalance{}
})
}

// Get the (account, asset) tuple from the Balances
// if the tuple is not present, it will write a big.NewInt(0) in it and return it
func (b Balances) fetchBalance(account string, asset string) *big.Int {
accountBalances := b.fetchAccountBalances(account)

return defaultMapGet(accountBalances, asset, func() *big.Int {
return new(big.Int)
})
}

func (b Balances) has(account string, asset string) bool {
accountBalances := defaultMapGet(b, account, func() AccountBalance {
return AccountBalance{}
})

_, ok := accountBalances[asset]
return ok
}

// given a BalanceQuery, return a new query which only contains needed (asset, account) pairs
// (that is, the ones that aren't already cached)
func (b Balances) filterQuery(q BalanceQuery) BalanceQuery {
filteredQuery := BalanceQuery{}
for accountName, queriedCurrencies := range q {
filteredCurrencies := utils.Filter(queriedCurrencies, func(currency string) bool {
return !b.has(accountName, currency)
})

if len(filteredCurrencies) > 0 {
filteredQuery[accountName] = filteredCurrencies
}

}
return filteredQuery
}

// Merge balances by adding balances in the "update" arg
func (b Balances) mergeBalance(update Balances) {
// merge queried balance
for acc, accBalances := range update {
cachedAcc := defaultMapGet(b, acc, func() AccountBalance {
return AccountBalance{}
})

Check warning on line 57 in internal/interpreter/balances.go

View check run for this annotation

Codecov / codecov/patch

internal/interpreter/balances.go#L56-L57

Added lines #L56 - L57 were not covered by tests

for curr, amt := range accBalances {
cachedAcc[curr] = amt
}
}
}
Comment on lines +52 to +63
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💡 Verification agent

🧩 Analysis chain

Effective balance merging implementation.

The mergeBalance method effectively combines balances from multiple sources, which is crucial for maintaining accurate state during script execution.

Note: Static analysis suggests lines 50-51 are not covered by tests. Consider adding a test case that exercises the scenario where a new account is added during a merge operation.


🏁 Script executed:

#!/bin/bash
# Check test coverage for the mergeBalance method, specifically for lines 50-51

# Find test files that might test the mergeBalance function
echo "Looking for tests that might cover the mergeBalance function..."
grep -r "mergeBalance" --include="*_test.go" .

Length of output: 183


Action: Add test coverage for the mergeBalance update scenario.

The current implementation in internal/interpreter/balances.go (lines 46-57) correctly merges balance updates; however, our grep check shows no test case covers the branch where a new account is introduced (specifically affecting lines 50-51). Please add a unit test that exercises this scenario to ensure that when an account does not exist in the cached balance, it is correctly initialized and updated.

  • File: internal/interpreter/balances.go (lines 46-57)
  • Concern: The merging logic for adding new accounts (lines 50-51) lacks test coverage.
  • Recommendation: Add a test case in the appropriate *_test.go file that simulates merging an update with a new account.
🧰 Tools
🪛 GitHub Check: codecov/patch

[warning] 50-51: internal/interpreter/balances.go#L50-L51
Added lines #L50 - L51 were not covered by tests

31 changes: 31 additions & 0 deletions internal/interpreter/balances_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
package interpreter

import (
"math/big"
"testing"

"github.com/stretchr/testify/require"
)

func TestFilterQuery(t *testing.T) {
fullBalance := Balances{
"alice": AccountBalance{
"EUR/2": big.NewInt(1),
"USD/2": big.NewInt(2),
},
"bob": AccountBalance{
"BTC": big.NewInt(3),
},
}

filteredQuery := fullBalance.filterQuery(BalanceQuery{
"alice": []string{"GBP/2", "YEN", "EUR/2"},
"bob": []string{"BTC"},
"charlie": []string{"ETH"},
})

require.Equal(t, BalanceQuery{
"alice": []string{"GBP/2", "YEN"},
"charlie": []string{"ETH"},
}, filteredQuery)
}
24 changes: 5 additions & 19 deletions internal/interpreter/batch_balances_query.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@ import (

"github.com/formancehq/numscript/internal/parser"
"github.com/formancehq/numscript/internal/utils"
"golang.org/x/exp/maps"
)

// traverse the script to batch in advance required balance queries
Expand Down Expand Up @@ -57,41 +56,28 @@ func (st *programState) batchQuery(account string, asset string) {
}

previousValues := st.CurrentBalanceQuery[account]
if !slices.Contains[[]string, string](previousValues, asset) {
if !slices.Contains(previousValues, asset) {
st.CurrentBalanceQuery[account] = append(previousValues, asset)
}
}

func (st *programState) runBalancesQuery() error {
filteredQuery := BalanceQuery{}
for accountName, queriedCurrencies := range st.CurrentBalanceQuery {

cachedCurrenciesForAccount := defaultMapGet(st.CachedBalances, accountName, func() AccountBalance {
return AccountBalance{}
})

for _, queriedCurrency := range queriedCurrencies {
isAlreadyCached := slices.Contains(maps.Keys(cachedCurrenciesForAccount), queriedCurrency)
if !isAlreadyCached {
filteredQuery[accountName] = queriedCurrencies
}
}

}
filteredQuery := st.CachedBalances.filterQuery(st.CurrentBalanceQuery)

// avoid updating balances if we don't need to fetch new data
if len(filteredQuery) == 0 {
return nil
}

balances, err := st.Store.GetBalances(st.ctx, filteredQuery)
queriedBalances, err := st.Store.GetBalances(st.ctx, filteredQuery)
if err != nil {
return err
}
// reset batch query
st.CurrentBalanceQuery = BalanceQuery{}

st.CachedBalances = balances
st.CachedBalances.mergeBalance(queriedBalances)

return nil
}

Expand Down
10 changes: 10 additions & 0 deletions internal/interpreter/evaluate_expr.go
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,16 @@ func (st *programState) evaluateExpr(expr parser.ValueExpr) (Value, InterpreterE
return nil, nil
}

case *parser.FnCall:
if !st.varOriginPosition {
err := st.checkFeatureFlag(ExperimentalMidScriptFunctionCall)
if err != nil {
return nil, err
}
}

return st.handleFnCall(nil, *expr)

default:
utils.NonExhaustiveMatchPanic[any](expr)
return nil, nil
Expand Down
Loading