From 163e60d6ed46f737f073add4ab7acd249430e5a3 Mon Sep 17 00:00:00 2001 From: Nicholas Cross Date: Sun, 23 Jun 2019 21:10:59 +1000 Subject: [PATCH] Lab 7 - Errors - Create an executable go program - Copy the CRUD puppy from upstream master - Add a custom error type `Error` with fields `Message` and `Code` - Extend the `Storer` interface for all methods to also return `error` - Create errors for: - Value < 0 - ID not found in Read, Update and Delete - Add locking for proper use of sync.Map --- 07_errors/nicholascross/main.go | 169 +++++++++++++++++++++++++++ 07_errors/nicholascross/main_test.go | 119 +++++++++++++++++++ 2 files changed, 288 insertions(+) create mode 100644 07_errors/nicholascross/main.go create mode 100644 07_errors/nicholascross/main_test.go diff --git a/07_errors/nicholascross/main.go b/07_errors/nicholascross/main.go new file mode 100644 index 000000000..649b65830 --- /dev/null +++ b/07_errors/nicholascross/main.go @@ -0,0 +1,169 @@ +package main + +import ( + "fmt" + "io" + "os" + "sync" +) + +type Puppy struct { + ID int64 + Breed string + Colour string + Value float64 +} + +type Storer interface { + CreatePuppy(p Puppy) (bool, error) + RetrievePuppy(int64) (*Puppy, error) + UpdatePuppy(int64, Puppy) (bool, error) + DeletePuppy(int64) (bool, error) +} + +type MapStore map[int64]Puppy + +const ( + missingPup = iota + invalidPupValue + pupAlreadyExists +) + +type PupError struct { + Message string + Code int +} + +func (e *PupError) Error() string { + return fmt.Sprintf("[%d] %s", e.Code, e.Message) +} + +func (e *PupError) IsMissingPup() bool { + return e.Code == missingPup +} + +func (e *PupError) PupAlreadyExists() bool { + return e.Code == pupAlreadyExists +} + +func (e *PupError) InvalidPupValue() bool { + return e.Code == invalidPupValue +} + +func missingPuppy(id int64) error { + return &PupError{fmt.Sprintf("Puppy not found: %d", id), missingPup} +} + +func puppyAlreadyExists(id int64) error { + return &PupError{fmt.Sprintf("Puppy already exists: %d", id), pupAlreadyExists} +} + +func invalidPuppyValue(value float64) error { + return &PupError{fmt.Sprintf("Puppy value must be non negative: %f", value), invalidPupValue} +} + +var out io.Writer = os.Stdout + +func main() { + p := Puppy{1, "Jack Russel Terrier", "white and brown", 550.0} + fmt.Fprintf(out, "%d - %s [%s]: %f", p.ID, p.Breed, p.Colour, p.Value) +} + +func (ms MapStore) CreatePuppy(p Puppy) (bool, error) { + if p.Value < 0 { + return false, invalidPuppyValue(p.Value) + } + + if _, ok := ms[p.ID]; ok { + return false, puppyAlreadyExists(p.ID) + } + + ms[p.ID] = p + return true, nil +} + +func (ms MapStore) RetrievePuppy(id int64) (*Puppy, error) { + if _, ok := ms[id]; ok { + pup := ms[id] + return &pup, nil + } + return nil, missingPuppy(id) +} + +func (ms MapStore) UpdatePuppy(id int64, p Puppy) (bool, error) { + if p.Value < 0 { + return false, invalidPuppyValue(p.Value) + } + + if _, ok := ms[id]; ok { + p.ID = id + ms[p.ID] = p + return true, nil + } + return false, missingPuppy(id) +} + +func (ms MapStore) DeletePuppy(id int64) (bool, error) { + if _, ok := ms[id]; ok { + delete(ms, id) + return true, nil + } + return false, missingPuppy(id) +} + +type SyncStore struct { + store *sync.Map + lock *sync.Mutex +} + +func (s SyncStore) CreatePuppy(p Puppy) (bool, error) { + if p.Value < 0 { + return false, invalidPuppyValue(p.Value) + } + + s.lock.Lock() + defer s.lock.Unlock() + + if _, ok := s.store.Load(p.ID); ok { + return false, puppyAlreadyExists(p.ID) + } + + s.store.Store(p.ID, p) + return true, nil +} + +func (s SyncStore) RetrievePuppy(id int64) (*Puppy, error) { + if pup, ok := s.store.Load(id); ok { + p := pup.(Puppy) + return &p, nil + } + return nil, missingPuppy(id) +} + +func (s SyncStore) UpdatePuppy(id int64, p Puppy) (bool, error) { + if p.Value < 0 { + return false, invalidPuppyValue(p.Value) + } + + s.lock.Lock() + defer s.lock.Unlock() + + if _, ok := s.store.Load(id); ok { + p.ID = id + s.store.Store(p.ID, p) + return true, nil + } + return false, missingPuppy(id) +} + +func (s SyncStore) DeletePuppy(id int64) (bool, error) { + + s.lock.Lock() + defer s.lock.Unlock() + + if _, ok := s.store.Load(id); ok { + s.store.Delete(id) + return true, nil + } + return false, missingPuppy(id) +} diff --git a/07_errors/nicholascross/main_test.go b/07_errors/nicholascross/main_test.go new file mode 100644 index 000000000..1837143c8 --- /dev/null +++ b/07_errors/nicholascross/main_test.go @@ -0,0 +1,119 @@ +package main + +import ( + "bytes" + "strconv" + "sync" + "testing" + + "github.com/stretchr/testify/suite" +) + +type StorerTestSuite struct { + suite.Suite + storer Storer +} + +func (suite *StorerTestSuite) SetupTest() { + ok, err := suite.storer.CreatePuppy(Puppy{1, "Jack Russel", "white", 500.0}) + if !ok && !err.(*PupError).PupAlreadyExists() { + suite.Fail("Failed to setup test suite") + } +} + +func (suite *StorerTestSuite) TestCreate() { + ok, _ := suite.storer.CreatePuppy(Puppy{12, "Komondor", "white", 1000.0}) + puppy, _ := suite.storer.RetrievePuppy(12) + suite.Equal(int64(12), puppy.ID) + suite.Equal("Komondor", puppy.Breed) + suite.Equal("white", puppy.Colour) + suite.Equal(1000.00, puppy.Value) + suite.Equal(true, ok) +} + +func (suite *StorerTestSuite) TestCreateInvalid() { + ok, err := suite.storer.CreatePuppy(Puppy{12, "Komondor", "white", -1000.0}) + suite.Equal(false, ok) + suite.Equal("[1] Puppy value must be non negative: -1000.000000", err.Error()) + suite.Equal(true, err.(*PupError).InvalidPupValue()) +} + +func (suite *StorerTestSuite) TestRead() { + puppy, _ := suite.storer.RetrievePuppy(1) + suite.Equal(int64(1), puppy.ID) + suite.Equal("Jack Russel", puppy.Breed) + suite.Equal("white", puppy.Colour) + suite.Equal(500.00, puppy.Value) +} + +func (suite *StorerTestSuite) TestUpdate() { + ok, _ := suite.storer.UpdatePuppy(1, Puppy{1, "Jack Russel Terrier", "white and brown", 550.0}) + puppy, _ := suite.storer.RetrievePuppy(1) + suite.Equal(int64(1), puppy.ID) + suite.Equal("Jack Russel Terrier", puppy.Breed) + suite.Equal("white and brown", puppy.Colour) + suite.Equal(550.00, puppy.Value) + suite.Equal(true, ok) +} + +func (suite *StorerTestSuite) TestUpdateMissing() { + ok, err := suite.storer.UpdatePuppy(10, Puppy{10, "Jack Russel Terrier", "white and brown", 550.0}) + suite.Equal(false, ok) + suite.Equal("[0] Puppy not found: 10", err.Error()) + suite.Equal(true, err.(*PupError).IsMissingPup()) +} + +func (suite *StorerTestSuite) TestUpdateInvalid() { + ok, err := suite.storer.UpdatePuppy(1, Puppy{1, "Jack Russel Terrier", "white and brown", -550.0}) + suite.Equal(false, ok) + suite.Equal("[1] Puppy value must be non negative: -550.000000", err.Error()) + suite.Equal(true, err.(*PupError).InvalidPupValue()) +} + +func (suite *StorerTestSuite) TestDelete() { + ok, _ := suite.storer.DeletePuppy(1) + puppy, _ := suite.storer.RetrievePuppy(1) + suite.Nil(puppy) + suite.Equal(true, ok) +} + +func (suite *StorerTestSuite) TestDeleteMissing() { + ok, err := suite.storer.DeletePuppy(10) + suite.Equal(false, ok) + suite.Equal("[0] Puppy not found: 10", err.Error()) + suite.Equal(true, err.(*PupError).IsMissingPup()) +} + +func TestMapStorer(t *testing.T) { + s := StorerTestSuite{ + storer: MapStore{}, + } + suite.Run(t, &s) +} + +func TestSyncStorer(t *testing.T) { + store := sync.Map{} + lock := sync.Mutex{} + + s := StorerTestSuite{ + storer: SyncStore{ + store: &store, + lock: &lock, + }, + } + suite.Run(t, &s) +} + +func TestMainOutput(t *testing.T) { + var buf bytes.Buffer + out = &buf + + main() + + expected := strconv.Quote("1 - Jack Russel Terrier [white and brown]: 550.000000") + actual := strconv.Quote(buf.String()) + + if expected != actual { + t.Errorf("Unexpected output. Expected: %q - Actual: %q", expected, actual) + } +}