-
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
11 changed files
with
288 additions
and
24 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,212 @@ | ||
package main | ||
|
||
import ( | ||
"fmt" | ||
"github.com/hovsep/fmesh" | ||
"github.com/hovsep/fmesh/component" | ||
"github.com/hovsep/fmesh/port" | ||
"github.com/hovsep/fmesh/signal" | ||
) | ||
|
||
type FactorizedNumber struct { | ||
Num int | ||
Factors []any | ||
} | ||
|
||
// This example demonstrates the ability to nest meshes, where a component within a mesh | ||
// can itself be another mesh. This nesting is recursive, allowing for an unlimited depth | ||
// of nested meshes. Each nested mesh behaves as an individual component within the larger | ||
// mesh, enabling complex and hierarchical workflows. | ||
// In this example we implement prime factorization (which is core part of RSA encryption algorithm) as a sub-mesh | ||
func main() { | ||
starter := component.New("starter"). | ||
WithDescription("This component just holds numbers we want to factorize"). | ||
WithInputs("in"). // Single port is enough, as it can hold any number of signals (as long as they fit into1 memory) | ||
WithOutputs("out"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
// Pure bypass | ||
return port.ForwardSignals(inputs.ByName("in"), outputs.ByName("out")) | ||
}) | ||
|
||
filter := component.New("filter"). | ||
WithDescription("In this component we can do some optional filtering"). | ||
WithInputs("in"). | ||
WithOutputs("out", "log"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
isValid := func(num int) bool { | ||
return num < 1000 | ||
} | ||
|
||
for _, sig := range inputs.ByName("in").AllSignalsOrNil() { | ||
if isValid(sig.PayloadOrNil().(int)) { | ||
outputs.ByName("out").PutSignals(sig) | ||
} else { | ||
outputs.ByName("log").PutSignals(sig) | ||
} | ||
} | ||
return nil | ||
}) | ||
|
||
logger := component.New("logger"). | ||
WithDescription("Simple logger"). | ||
WithInputs("in"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
log := func(line string) { | ||
fmt.Printf("LOG: %s", line) | ||
} | ||
|
||
for _, sig := range inputs.ByName("in").AllSignalsOrNil() { | ||
if logLine := sig.PayloadOrNil(); logLine != nil { | ||
log(logLine.(string)) | ||
} | ||
} | ||
return nil | ||
}) | ||
|
||
factorizer := component.New("factorizer"). | ||
WithDescription("Prime factorization implemented as separate f-mesh"). | ||
WithInputs("in"). | ||
WithOutputs("out"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
//This activation function has no implementation of factorization algorithm, | ||
//it only runs another f-mesh to get results | ||
|
||
//Get nested mesh or meshes | ||
factorization := getPrimeFactorizationMesh() | ||
|
||
// As nested f-mesh processes 1 signal per run we run it in the loop per each number | ||
for _, numSig := range inputs.ByName("in").AllSignalsOrNil() { | ||
//Set init data to nested mesh (pass signals from outer mesh to inner one) | ||
factorization.Components().ByName("starter").InputByName("in").PutSignals(numSig) | ||
|
||
//Run nested mesh | ||
_, err := factorization.Run() | ||
|
||
if err != nil { | ||
return fmt.Errorf("inner mesh failed: %w", err) | ||
} | ||
|
||
// Get results from nested mesh | ||
factors, err := factorization.Components().ByName("results").OutputByName("factors").AllSignalsPayloads() | ||
if err != nil { | ||
return fmt.Errorf("failed to get factors: %w", err) | ||
} | ||
|
||
//Pass results to outer mesh | ||
number := numSig.PayloadOrNil().(int) | ||
outputs.ByName("out").PutSignals(signal.New(FactorizedNumber{ | ||
Num: number, | ||
Factors: factors, | ||
})) | ||
} | ||
|
||
return nil | ||
}) | ||
|
||
//Setup pipes | ||
starter.OutputByName("out").PipeTo(filter.InputByName("in")) | ||
filter.OutputByName("log").PipeTo(logger.InputByName("in")) | ||
filter.OutputByName("out").PipeTo(factorizer.InputByName("in")) | ||
|
||
// Build the mesh | ||
outerMesh := fmesh.New("outer").WithComponents(starter, filter, logger, factorizer) | ||
|
||
//Set init data | ||
outerMesh.Components(). | ||
ByName("starter"). | ||
InputByName("in"). | ||
PutSignals(signal.NewGroup(315).SignalsOrNil()...) | ||
|
||
//Run outer mesh | ||
_, err := outerMesh.Run() | ||
|
||
if err != nil { | ||
fmt.Println(fmt.Errorf("outer mesh failed with error: %w", err)) | ||
} | ||
|
||
//Read results | ||
for _, resSig := range outerMesh.Components().ByName("factorizer").OutputByName("out").AllSignalsOrNil() { | ||
result := resSig.PayloadOrNil().(FactorizedNumber) | ||
fmt.Println(fmt.Sprintf("Factors of number %d : %v", result.Num, result.Factors)) | ||
} | ||
} | ||
|
||
func getPrimeFactorizationMesh() *fmesh.FMesh { | ||
starter := component.New("starter"). | ||
WithDescription("Load the number to be factorized"). | ||
WithInputs("in"). | ||
WithOutputs("out"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
//For simplicity this f-mesh processes only one signal per run, so ignore all except first | ||
outputs.ByName("out").PutSignals(inputs.ByName("in").Buffer().First()) | ||
return nil | ||
}) | ||
|
||
d2 := component.New("d2"). | ||
WithDescription("Divide by smallest prime (2) to handle even factors"). | ||
WithInputs("in"). | ||
WithOutputs("out", "factor"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
number := inputs.ByName("in").FirstSignalPayloadOrNil().(int) | ||
|
||
for number%2 == 0 { | ||
outputs.ByName("factor").PutSignals(signal.New(2)) | ||
number /= 2 | ||
} | ||
|
||
outputs.ByName("out").PutSignals(signal.New(number)) | ||
return nil | ||
}) | ||
|
||
dodd := component.New("dodd"). | ||
WithDescription("Divide by odd primes starting from 3"). | ||
WithInputs("in"). | ||
WithOutputs("out", "factor"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
number := inputs.ByName("in").FirstSignalPayloadOrNil().(int) | ||
divisor := 3 | ||
for number > 1 && divisor*divisor <= number { | ||
for number%divisor == 0 { | ||
outputs.ByName("factor").PutSignals(signal.New(divisor)) | ||
number /= divisor | ||
} | ||
divisor += 2 | ||
} | ||
outputs.ByName("out").PutSignals(signal.New(number)) | ||
return nil | ||
}) | ||
|
||
finalPrime := component.New("final_prime"). | ||
WithDescription("Store the last remaining prime factor, if any"). | ||
WithInputs("in"). | ||
WithOutputs("factor"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
number := inputs.ByName("in").FirstSignalPayloadOrNil().(int) | ||
if number > 1 { | ||
outputs.ByName("factor").PutSignals(signal.New(number)) | ||
} | ||
return nil | ||
}) | ||
|
||
results := component.New("results"). | ||
WithDescription("factors holder"). | ||
WithInputs("factor"). | ||
WithOutputs("factors"). | ||
WithActivationFunc(func(inputs *port.Collection, outputs *port.Collection) error { | ||
return port.ForwardSignals(inputs.ByName("factor"), outputs.ByName("factors")) | ||
}) | ||
|
||
//Main pipeline starter->d2->dodd->finalPrime | ||
starter.OutputByName("out").PipeTo(d2.InputByName("in")) | ||
d2.OutputByName("out").PipeTo(dodd.InputByName("in")) | ||
dodd.OutputByName("out").PipeTo(finalPrime.InputByName("in")) | ||
|
||
//All found factors are accumulated in results | ||
d2.OutputByName("factor").PipeTo(results.InputByName("factor")) | ||
dodd.OutputByName("factor").PipeTo(results.InputByName("factor")) | ||
finalPrime.OutputByName("factor").PipeTo(results.InputByName("factor")) | ||
|
||
return fmesh.New("prime factors algo"). | ||
WithDescription("Pass single signal to starter"). | ||
WithComponents(starter, d2, dodd, finalPrime, results) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.