Skip to content

Commit

Permalink
Royale mode does damage instead of eliminating.
Browse files Browse the repository at this point in the history
  • Loading branch information
bvanvugt committed Jul 27, 2020
1 parent dd5a2fd commit 2ca57f0
Show file tree
Hide file tree
Showing 2 changed files with 111 additions and 20 deletions.
41 changes: 31 additions & 10 deletions royale.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,48 +9,61 @@ type RoyaleRuleset struct {

Turn int32
ShrinkEveryNTurns int32
DamagePerTurn int32

// Output
OutOfBounds []Point
}

func (r *RoyaleRuleset) CreateNextBoardState(prevState *BoardState, moves []SnakeMove) (*BoardState, error) {
if r.ShrinkEveryNTurns < 1 {
return nil, errors.New("royale game must shrink at least every 1 turn")
return nil, errors.New("royale game must shrink at least every turn")
}

nextBoardState, err := r.StandardRuleset.CreateNextBoardState(prevState, moves)
if err != nil {
return nil, err
}

// Algorithm:
// - Populate OOB for last turn
// - Apply damage to snake heads that are OOB
// - Re-populate OOB for this turn
// ---> This means damage on board shrinks doesn't hit until the following turn.

// TODO: LOG?
err = r.populateOutOfBounds(nextBoardState, r.Turn-1)
if err != nil {
return nil, err
}

// TODO: LOG?
err = r.populateOutOfBounds(nextBoardState)
err = r.damageOutOfBounds(nextBoardState)
if err != nil {
return nil, err
}

// TODO: LOG?
err = r.eliminateOutOfBounds(nextBoardState)
err = r.populateOutOfBounds(nextBoardState, r.Turn)
if err != nil {
return nil, err
}

return nextBoardState, nil
}

func (r *RoyaleRuleset) populateOutOfBounds(b *BoardState) error {
func (r *RoyaleRuleset) populateOutOfBounds(b *BoardState, turn int32) error {
r.OutOfBounds = []Point{}

if r.ShrinkEveryNTurns < 1 {
return errors.New("royale game must shrink at least every 1 turn")
return errors.New("royale game must shrink at least every turn")
}

if r.Turn < r.ShrinkEveryNTurns {
if turn < r.ShrinkEveryNTurns {
return nil
}

numShrinks := r.Turn / r.ShrinkEveryNTurns
numShrinks := turn / r.ShrinkEveryNTurns
minX, maxX := numShrinks, b.Width-1-numShrinks
minY, maxY := numShrinks, b.Height-1-numShrinks
for x := int32(0); x < b.Width; x++ {
Expand All @@ -64,15 +77,23 @@ func (r *RoyaleRuleset) populateOutOfBounds(b *BoardState) error {
return nil
}

func (r *RoyaleRuleset) eliminateOutOfBounds(b *BoardState) error {
func (r *RoyaleRuleset) damageOutOfBounds(b *BoardState) error {
if r.DamagePerTurn < 1 {
return errors.New("royale damage per turn must be greater than zero")
}

for i := 0; i < len(b.Snakes); i++ {
snake := &b.Snakes[i]
if snake.EliminatedCause == NotEliminated {
head := snake.Body[0]
for _, p := range r.OutOfBounds {
if head == p {
// Snake is now out of bounds, eliminate it
snake.EliminatedCause = EliminatedByOutOfBounds
// Snake is now out of bounds, reduce health
snake.Health = snake.Health - r.DamagePerTurn
if snake.Health <= 0 {
snake.Health = 0
snake.EliminatedCause = EliminatedByStarvation
}
}
}
}
Expand Down
90 changes: 80 additions & 10 deletions royale_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,14 +16,14 @@ func TestRoyaleDefaultSanity(t *testing.T) {
r := RoyaleRuleset{}
_, err := r.CreateNextBoardState(boardState, []SnakeMove{})
require.Error(t, err)
require.Equal(t, err, errors.New("royale game must shrink at least every 1 turn"))
require.Equal(t, errors.New("royale game must shrink at least every turn"), err)

r = RoyaleRuleset{ShrinkEveryNTurns: 1}
r = RoyaleRuleset{ShrinkEveryNTurns: 1, DamagePerTurn: 1}
_, err = r.CreateNextBoardState(boardState, []SnakeMove{})
require.NoError(t, err)
}

func TestRoyalePopulateObstacles(t *testing.T) {
func TestRoyaleOutOfBounds(t *testing.T) {
tests := []struct {
Width int32
Height int32
Expand All @@ -32,7 +32,7 @@ func TestRoyalePopulateObstacles(t *testing.T) {
Error error
ExpectedOutOfBounds []Point
}{
{Error: errors.New("royale game must shrink at least every 1 turn")},
{Error: errors.New("royale game must shrink at least every turn")},
{ShrinkEveryNTurns: 1, ExpectedOutOfBounds: []Point{}},
{Turn: 1, ShrinkEveryNTurns: 1, ExpectedOutOfBounds: []Point{}},
{Width: 3, Height: 3, Turn: 1, ShrinkEveryNTurns: 10, ExpectedOutOfBounds: []Point{}},
Expand Down Expand Up @@ -62,7 +62,7 @@ func TestRoyalePopulateObstacles(t *testing.T) {
ShrinkEveryNTurns: test.ShrinkEveryNTurns,
}

err := r.populateOutOfBounds(b)
err := r.populateOutOfBounds(b, test.Turn)
require.Equal(t, test.Error, err)
if err == nil {
// Obstacles should match
Expand All @@ -81,7 +81,7 @@ func TestRoyalePopulateObstacles(t *testing.T) {
}
}

func TestRoyaleEliminateOutOfBounds(t *testing.T) {
func TestRoyaleDamageOutOfBounds(t *testing.T) {
tests := []struct {
Snakes []Snake
OutOfBounds []Point
Expand All @@ -98,7 +98,7 @@ func TestRoyaleEliminateOutOfBounds(t *testing.T) {
{
Snakes: []Snake{{Body: []Point{{0, 0}}}},
OutOfBounds: []Point{{0, 0}},
ExpectedEliminatedCauses: []string{EliminatedByOutOfBounds},
ExpectedEliminatedCauses: []string{EliminatedByStarvation},
ExpectedEliminatedByIDs: []string{""},
},
{
Expand All @@ -122,15 +122,15 @@ func TestRoyaleEliminateOutOfBounds(t *testing.T) {
{Body: []Point{{3, 3}, {3, 4}, {3, 5}, {3, 6}}},
},
OutOfBounds: []Point{{3, 3}},
ExpectedEliminatedCauses: []string{NotEliminated, EliminatedByOutOfBounds},
ExpectedEliminatedCauses: []string{NotEliminated, EliminatedByStarvation},
ExpectedEliminatedByIDs: []string{"", ""},
},
}

for _, test := range tests {
b := &BoardState{Snakes: test.Snakes}
r := RoyaleRuleset{OutOfBounds: test.OutOfBounds}
err := r.eliminateOutOfBounds(b)
r := RoyaleRuleset{OutOfBounds: test.OutOfBounds, DamagePerTurn: 100}
err := r.damageOutOfBounds(b)
require.NoError(t, err)

for i, snake := range b.Snakes {
Expand All @@ -139,3 +139,73 @@ func TestRoyaleEliminateOutOfBounds(t *testing.T) {

}
}

func TestRoyaleDamagePerTurn(t *testing.T) {
tests := []struct {
Health int32
DamagePerTurn int32
ExpectedHealth int32
ExpectedEliminationCause string
Error error
}{
{100, 0, 100, NotEliminated, errors.New("royale damage per turn must be greater than zero")},
{100, -100, 100, NotEliminated, errors.New("royale damage per turn must be greater than zero")},
{100, 1, 99, NotEliminated, nil},
{100, 99, 1, NotEliminated, nil},
{100, 100, 0, EliminatedByStarvation, nil},
{100, 101, 0, EliminatedByStarvation, nil},
{100, 999, 0, EliminatedByStarvation, nil},
{2, 1, 1, NotEliminated, nil},
{1, 1, 0, EliminatedByStarvation, nil},
{1, 999, 0, EliminatedByStarvation, nil},
{0, 1, 0, EliminatedByStarvation, nil},
{0, 999, 0, EliminatedByStarvation, nil},
}

for _, test := range tests {
b := &BoardState{Snakes: []Snake{{Health: test.Health, Body: []Point{{0, 0}}}}}
r := RoyaleRuleset{OutOfBounds: []Point{{0, 0}}, DamagePerTurn: test.DamagePerTurn}

err := r.damageOutOfBounds(b)
require.Equal(t, test.Error, err)
require.Equal(t, test.ExpectedHealth, b.Snakes[0].Health)
require.Equal(t, test.ExpectedEliminationCause, b.Snakes[0].EliminatedCause)
}
}

func TestRoyalDamageNextTurn(t *testing.T) {
b := &BoardState{Width: 10, Height: 10, Snakes: []Snake{{ID: "one", Health: 100, Body: []Point{{1, 1}}}}}
r := RoyaleRuleset{Turn: 10, ShrinkEveryNTurns: 10, DamagePerTurn: 30}
m := []SnakeMove{{ID: "one", Move: "right"}}

n, err := r.CreateNextBoardState(b, m)
require.NoError(t, err)
require.Equal(t, NotEliminated, n.Snakes[0].EliminatedCause)
require.Equal(t, int32(99), n.Snakes[0].Health)
require.Equal(t, Point{2, 1}, n.Snakes[0].Body[0])
require.Equal(t, 36, len(r.OutOfBounds))

r.Turn = 20
n, err = r.CreateNextBoardState(b, m)
require.NoError(t, err)
require.Equal(t, NotEliminated, n.Snakes[0].EliminatedCause)
require.Equal(t, int32(99), n.Snakes[0].Health)
require.Equal(t, Point{2, 1}, n.Snakes[0].Body[0])
require.Equal(t, 64, len(r.OutOfBounds))

r.Turn = 21
n, err = r.CreateNextBoardState(b, m)
require.NoError(t, err)
require.Equal(t, NotEliminated, n.Snakes[0].EliminatedCause)
require.Equal(t, int32(69), n.Snakes[0].Health)
require.Equal(t, Point{2, 1}, n.Snakes[0].Body[0])
require.Equal(t, 64, len(r.OutOfBounds))

b.Snakes[0].Health = 15
n, err = r.CreateNextBoardState(b, m)
require.NoError(t, err)
require.Equal(t, EliminatedByStarvation, n.Snakes[0].EliminatedCause)
require.Equal(t, int32(0), n.Snakes[0].Health)
require.Equal(t, Point{2, 1}, n.Snakes[0].Body[0])
require.Equal(t, 64, len(r.OutOfBounds))
}

0 comments on commit 2ca57f0

Please sign in to comment.