|
@@ -51,11 +51,14 @@ type SnakeSimulator struct {
|
|
|
remoteControl *remotecontrol.RemoteControl
|
|
|
|
|
|
//GUI interface part
|
|
|
- speed uint32
|
|
|
- fieldUpdateQueue chan bool
|
|
|
- snakeUpdateQueue chan bool
|
|
|
- statsUpdateQueue chan bool
|
|
|
- speedQueue chan uint32
|
|
|
+ speed uint32
|
|
|
+ fieldUpdateQueue chan bool
|
|
|
+ snakeUpdateQueue chan bool
|
|
|
+ statsUpdateQueue chan bool
|
|
|
+ isPlayingUpdateQueue chan bool
|
|
|
+ speedQueue chan uint32
|
|
|
+ isPlaying bool
|
|
|
+ repeatInLoop bool
|
|
|
|
|
|
snakeReadMutex sync.Mutex
|
|
|
fieldReadMutex sync.Mutex
|
|
@@ -81,6 +84,7 @@ func NewSnakeSimulator(maxVerificationSteps int) (s *SnakeSimulator) {
|
|
|
fieldUpdateQueue: make(chan bool, 2),
|
|
|
snakeUpdateQueue: make(chan bool, 2),
|
|
|
statsUpdateQueue: make(chan bool, 2),
|
|
|
+ isPlayingUpdateQueue: make(chan bool, 1),
|
|
|
speedQueue: make(chan uint32, 1),
|
|
|
speed: 10,
|
|
|
remoteControl: remotecontrol.NewRemoteControl(),
|
|
@@ -125,13 +129,51 @@ func (s *SnakeSimulator) Verify(population *genetic.Population) (fitnesses []*ge
|
|
|
s.fieldUpdateQueue <- true
|
|
|
prevSpeed := s.speed
|
|
|
s.speed = 5
|
|
|
- population.Networks[fitnesses[0].Index].SetStateWatcher(s.remoteControl)
|
|
|
- s.runSnake(population.Networks[fitnesses[0].Index], false)
|
|
|
- population.Networks[fitnesses[0].Index].SetStateWatcher(nil)
|
|
|
+ if s.isPlaying == true {
|
|
|
+ // Play best of the best
|
|
|
+ s.isPlaying = false
|
|
|
+ s.isPlayingUpdateQueue <- s.isPlaying
|
|
|
+ population.GetBestNetwork().SetStateWatcher(s.remoteControl)
|
|
|
+ s.runSnake(population.GetBestNetwork(), false)
|
|
|
+ population.GetBestNetwork().SetStateWatcher(nil)
|
|
|
+ } else {
|
|
|
+ // Pley best from generation
|
|
|
+ population.Networks[fitnesses[0].Index].SetStateWatcher(s.remoteControl)
|
|
|
+ s.runSnake(population.Networks[fitnesses[0].Index], false)
|
|
|
+ population.Networks[fitnesses[0].Index].SetStateWatcher(nil)
|
|
|
+ }
|
|
|
s.speed = prevSpeed
|
|
|
return
|
|
|
}
|
|
|
|
|
|
+func (s *SnakeSimulator) PlayBestNetwork(network *neuralnetwork.NeuralNetwork) {
|
|
|
+
|
|
|
+ for s.repeatInLoop {
|
|
|
+ s.remoteControl.Init(network)
|
|
|
+ s.stats.Generation++
|
|
|
+ s.statsUpdateQueue <- true
|
|
|
+
|
|
|
+ s.field.GenerateNextFood()
|
|
|
+ if s.speed > 0 {
|
|
|
+ s.fieldUpdateQueue <- true
|
|
|
+ }
|
|
|
+
|
|
|
+ //Best snake showtime!
|
|
|
+ s.fieldReadMutex.Lock()
|
|
|
+ s.field.GenerateNextFood()
|
|
|
+ s.fieldReadMutex.Unlock()
|
|
|
+ s.fieldUpdateQueue <- true
|
|
|
+ s.isPlaying = false
|
|
|
+ s.isPlayingUpdateQueue <- s.isPlaying
|
|
|
+ prevSpeed := s.speed
|
|
|
+ s.speed = 5
|
|
|
+ network.SetStateWatcher(s.remoteControl)
|
|
|
+ s.runSnake(network, false)
|
|
|
+ network.SetStateWatcher(nil)
|
|
|
+ s.speed = prevSpeed
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
func (s *SnakeSimulator) runSnake(inidividual *neuralnetwork.NeuralNetwork, randomStart bool) {
|
|
|
s.snakeReadMutex.Lock()
|
|
|
if randomStart {
|
|
@@ -447,3 +489,32 @@ func (s *SnakeSimulator) SetSpeed(ctx context.Context, speed *Speed) (*None, err
|
|
|
s.speedQueue <- speed.Speed
|
|
|
return &None{}, nil
|
|
|
}
|
|
|
+
|
|
|
+// Ask to play requested from gRPC GUI client
|
|
|
+func (s *SnakeSimulator) PlayBest(ctx context.Context, _ *None) (*None, error) {
|
|
|
+ s.isPlaying = true
|
|
|
+ s.isPlayingUpdateQueue <- s.isPlaying
|
|
|
+ return &None{}, nil
|
|
|
+}
|
|
|
+
|
|
|
+// Play in loop
|
|
|
+func (s *SnakeSimulator) PlayBestInLoop(_ context.Context, playBest *PlayingBestState) (*None, error) {
|
|
|
+ s.repeatInLoop = playBest.State
|
|
|
+ return &None{}, nil
|
|
|
+}
|
|
|
+
|
|
|
+// State of playing
|
|
|
+func (s *SnakeSimulator) IsPlaying(_ *None, srv SnakeSimulator_IsPlayingServer) error {
|
|
|
+ ctx := srv.Context()
|
|
|
+ for {
|
|
|
+ select {
|
|
|
+ case <-ctx.Done():
|
|
|
+ return ctx.Err()
|
|
|
+ default:
|
|
|
+ }
|
|
|
+ srv.Send(&PlayingBestState{
|
|
|
+ State: s.isPlaying,
|
|
|
+ })
|
|
|
+ <-s.isPlayingUpdateQueue
|
|
|
+ }
|
|
|
+}
|