// VulcanizeDB
// Copyright © 2019 Vulcanize

// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.

// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU Affero General Public License for more details.

// You should have received a copy of the GNU Affero General Public License
// along with this program.  If not, see <http://www.gnu.org/licenses/>.

package history_test

import (
	"math/big"

	. "github.com/onsi/ginkgo"
	. "github.com/onsi/gomega"
	"github.com/vulcanize/vulcanizedb/pkg/eth/fakes"
	"github.com/vulcanize/vulcanizedb/pkg/eth/history"
)

var _ = Describe("Populating blocks", func() {
	var blockRepository *fakes.MockBlockRepository

	BeforeEach(func() {
		blockRepository = fakes.NewMockBlockRepository()
	})

	It("fills in the only missing block (BlockNumber 1)", func() {
		blockChain := fakes.NewMockBlockChain()
		blockChain.SetLastBlock(big.NewInt(2))
		blockRepository.SetMissingBlockNumbersReturnArray([]int64{2})

		blocksAdded, err := history.PopulateMissingBlocks(blockChain, blockRepository, 1)
		Expect(err).NotTo(HaveOccurred())
		_, err = blockRepository.GetBlock(1)

		Expect(blocksAdded).To(Equal(1))
		Expect(err).ToNot(HaveOccurred())
	})

	It("fills in the three missing blocks (Numbers: 5,8,10)", func() {
		blockChain := fakes.NewMockBlockChain()
		blockChain.SetLastBlock(big.NewInt(13))
		blockRepository.SetMissingBlockNumbersReturnArray([]int64{5, 8, 10})

		blocksAdded, err := history.PopulateMissingBlocks(blockChain, blockRepository, 5)

		Expect(err).NotTo(HaveOccurred())
		Expect(blocksAdded).To(Equal(3))
		blockRepository.AssertCreateOrUpdateBlocksCallCountAndBlockNumbersEquals(3, []int64{5, 8, 10})
	})

	It("returns the number of blocks created", func() {
		blockChain := fakes.NewMockBlockChain()
		blockChain.SetLastBlock(big.NewInt(6))
		blockRepository.SetMissingBlockNumbersReturnArray([]int64{4, 5})

		numberOfBlocksCreated, err := history.PopulateMissingBlocks(blockChain, blockRepository, 3)

		Expect(err).NotTo(HaveOccurred())
		Expect(numberOfBlocksCreated).To(Equal(2))
	})

	It("updates the repository with a range of blocks w/in the range ", func() {
		blockChain := fakes.NewMockBlockChain()

		_, err := history.RetrieveAndUpdateBlocks(blockChain, blockRepository, history.MakeRange(2, 5))

		Expect(err).NotTo(HaveOccurred())
		blockRepository.AssertCreateOrUpdateBlocksCallCountAndBlockNumbersEquals(4, []int64{2, 3, 4, 5})
	})

	It("does not call repository create block when there is an error", func() {
		blockChain := fakes.NewMockBlockChain()
		blockChain.SetGetBlockByNumberErr(fakes.FakeError)
		blocks := history.MakeRange(1, 10)

		_, err := history.RetrieveAndUpdateBlocks(blockChain, blockRepository, blocks)

		Expect(err).To(HaveOccurred())
		blockRepository.AssertCreateOrUpdateBlockCallCountEquals(0)
	})
})