examples/advanced/05_nd_recursion.cat
#!/usr/bin/env catnip
# RUN: catnip docs/examples/advanced/05_nd_recursion.cat
# ND-Recursion : syntaxe des opérateurs
#
# Ces exemples illustrent la forme implémentée.

b = import('builtins')
is_list = (x) => { b.isinstance(x, b.list) }

# ~~ : ND-recursion

# Forme combinateur : seed initial + lambda avec recur
~~(0, (v, recur) => {
        if v < 10 { recur(v + 1) }
        else { v }
    })

# Forme déclaration : fonction ND-recursive
countdown = ~~(n, recur) => {
    if n > 0 { recur(n - 1) }
    else { "done" }
}

# Forme broadcast simple (equivalent a map)
data = list(1, 2, 3)
data.[~~(v, _) => { v * 2 }]

# Forme broadcast récursive (traverse les structures)
nested = list(1, list(2, 3), 4)
nested.[~~(v, recur) => {
        if is_list(v) { v.[~>(x) => { recur(x) }] }
        else { v * 2 }
    }]

# ~> : ND-map (lift)

# Lift une fonction dans le contexte ND
lifted_abs = ~> abs

# Forme broadcast : map en contexte ND
data.[~> abs]

# ~[] : topos vide

# Element neutre des operations ND
empty = ~[]

# Modes d'exécution

# Mode sequential (défaut) : debug, petits calculs
pragma('nd_mode', ND.sequential)
~~(10, (n, r) => { if n <= 1 { 1 } else { n * r(n - 1) } })

# Mode threads : memoization partagee, I/O bound
pragma('nd_mode', ND.thread)
pragma('nd_memoize', True)
~~(20, (n, r) => { if n <= 1 { n } else { r(n - 1) + r(n - 2) } })

# Mode processes : vrai parallelisme, CPU bound
pragma('nd_mode', ND.process)
pragma('nd_workers', 4)
list(5, 6, 7, 8).[~~(n, r) => {
        if n <= 1 { 1 }
        else { n * r(n - 1) }
    }]