Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

12 add api documentation #7

Open
wants to merge 87 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 2 commits
Commits
Show all changes
87 commits
Select commit Hold shift + click to select a range
4d7a7dd
remove core limitation
matheuspolitano Dec 23, 2024
6c29e2f
fix main bugs
matheuspolitano Dec 24, 2024
1601264
fix main bugs
matheuspolitano Dec 24, 2024
1310b7b
down go version
matheuspolitano Dec 24, 2024
aa08816
add release file
matheuspolitano Dec 24, 2024
8e08e34
add release info in readme
matheuspolitano Dec 24, 2024
70f5b75
Merge pull request #1 from matheuspolitano/1-fix-main-bugs
matheuspolitano Dec 24, 2024
0982ac0
add new pipeline
matheuspolitano Dec 24, 2024
011b3a8
Merge pull request #2 from matheuspolitano/1-fix-main-bugs-1
matheuspolitano Dec 24, 2024
fee950d
fix pipeline
matheuspolitano Dec 24, 2024
6df7c14
Merge pull request #3 from matheuspolitano/1-fix-main-bugs-2
matheuspolitano Dec 24, 2024
3690243
Add new release file release_20241224104515.toml
github-actions[bot] Dec 24, 2024
4285907
improve the pipeline
matheuspolitano Dec 24, 2024
015183f
update release
matheuspolitano Dec 24, 2024
9a12574
Merge pull request #4 from matheuspolitano/2-improve-the-pipeline
matheuspolitano Dec 24, 2024
e8914ac
Add new release file release_20241224105329.toml
github-actions[bot] Dec 24, 2024
a363d78
pipeline adjust
matheuspolitano Dec 24, 2024
c9c86ad
Merge pull request #5 from matheuspolitano/3-remove-release-file
matheuspolitano Dec 24, 2024
d51f15e
Add new release file release_20241224111005.toml
github-actions[bot] Dec 24, 2024
d766a84
improve errors messages
matheuspolitano Dec 24, 2024
375a284
ädd local env and extras check in config
matheuspolitano Dec 24, 2024
a4e8ec9
change main go file
matheuspolitano Dec 24, 2024
497f3c8
add config
matheuspolitano Dec 24, 2024
4b0ed73
add config test
matheuspolitano Dec 24, 2024
e904542
run in sequence de config test and improve the config
matheuspolitano Dec 24, 2024
29e68c4
fix type error in port
matheuspolitano Dec 24, 2024
314338f
add note in release
matheuspolitano Dec 24, 2024
90991ca
improve task detail
matheuspolitano Dec 24, 2024
858c708
Merge pull request #6 from matheuspolitano/5-improve-config
matheuspolitano Dec 24, 2024
55ba43f
Add new release file release_20241224141902.toml
github-actions[bot] Dec 24, 2024
8a9556d
update readme
matheuspolitano Dec 24, 2024
049a9af
update readme
matheuspolitano Dec 24, 2024
0a5b0ce
update readme
matheuspolitano Dec 24, 2024
be3a5b7
update readme
matheuspolitano Dec 24, 2024
76b5c7f
add myduration
matheuspolitano Dec 25, 2024
6fc90b0
add test
matheuspolitano Dec 25, 2024
29092dc
add release
matheuspolitano Dec 25, 2024
ecb6071
Merge pull request #8 from matheuspolitano/7-improve-time-alive
matheuspolitano Dec 25, 2024
3e48f98
Add new release file release_20241225114904.toml
github-actions[bot] Dec 25, 2024
650acee
improve main.go and server start and shutdown
matheuspolitano Dec 25, 2024
dadf977
improve readme
matheuspolitano Dec 25, 2024
37eecea
Merge pull request #9 from matheuspolitano/6-improve-main.go
matheuspolitano Dec 25, 2024
81a767c
Add new release file release_20241225141728.toml
github-actions[bot] Dec 25, 2024
f6d55be
add env as type
matheuspolitano Dec 25, 2024
c99942f
add new pacakge and set new test
matheuspolitano Dec 25, 2024
d7b3d96
add production mod
matheuspolitano Dec 25, 2024
a94b47f
remove useless comments
matheuspolitano Dec 25, 2024
798ec5a
add max age
matheuspolitano Dec 25, 2024
e2dfaa5
improve response
matheuspolitano Dec 25, 2024
aa4a2f6
improve config and add new config
matheuspolitano Dec 25, 2024
1e16f01
update dockerimage
matheuspolitano Dec 25, 2024
4c3e893
add in readme
matheuspolitano Dec 25, 2024
2ad6d1f
add release
matheuspolitano Dec 25, 2024
aa6389b
update go version
matheuspolitano Dec 25, 2024
d355e9b
update pipeline and go version
matheuspolitano Dec 25, 2024
e1b534d
ignore scripts
matheuspolitano Dec 26, 2024
57f302b
improve release pipeline
matheuspolitano Dec 26, 2024
215b9c5
add ignore
matheuspolitano Dec 26, 2024
6a874e9
improve the validate release
matheuspolitano Dec 26, 2024
088cedf
improve release
matheuspolitano Dec 26, 2024
da84809
add response
matheuspolitano Dec 26, 2024
bdc9bbe
improve format
matheuspolitano Dec 26, 2024
49429e7
update release
matheuspolitano Dec 26, 2024
79b53c0
fix test
matheuspolitano Dec 26, 2024
c2db65b
Merge pull request #10 from matheuspolitano/7-improve-gin-api
matheuspolitano Dec 26, 2024
0b2c8a2
Add new release file release_20241226171013.toml
github-actions[bot] Dec 26, 2024
f203a84
improve memdb
matheuspolitano Dec 26, 2024
1cea555
Merge pull request #11 from matheuspolitano/8-improve-repository
matheuspolitano Dec 26, 2024
a8acdba
Add new release file release_20241226174317.toml
github-actions[bot] Dec 26, 2024
caa30ad
add docker
matheuspolitano Dec 26, 2024
69069d6
add release
matheuspolitano Dec 26, 2024
aad42e9
add release
matheuspolitano Dec 26, 2024
80eccc2
Merge pull request #12 from matheuspolitano/9-setup-docker
matheuspolitano Dec 27, 2024
c245e41
Release: Add release_20241227000208.toml
github-actions[bot] Dec 27, 2024
64c930a
cicd adjust
matheuspolitano Dec 27, 2024
ebc0cf5
Merge pull request #13 from matheuspolitano/10-migrate-deploy-file
matheuspolitano Dec 27, 2024
dedcac4
Release: Add release_20241227122723.toml
github-actions[bot] Dec 27, 2024
5166895
11 improve service #major (#14)
matheuspolitano Dec 27, 2024
06ac48a
Release: Add release_20241227124702.toml
github-actions[bot] Dec 27, 2024
cde0067
add doc
matheuspolitano Dec 27, 2024
e5c9e4b
improve readme
matheuspolitano Dec 27, 2024
9fa2a51
improve readme
matheuspolitano Dec 27, 2024
12d5bb3
improve readme
matheuspolitano Dec 27, 2024
10af987
improve readme
matheuspolitano Dec 27, 2024
89cbe2a
improve readme
matheuspolitano Dec 27, 2024
94d8b9a
add latest push in docker
matheuspolitano Dec 27, 2024
5704f8e
go mod tidy
matheuspolitano Dec 27, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion cmd/coupon_service/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ func main() {
log.Fatalf("Failed to load config: %v", err)
}

repo := memdb.New()
repo := memdb.NewRepository()
svc := service.New(repo)

server, err := api.New(cfg.API, svc)
Expand Down
84 changes: 67 additions & 17 deletions internal/repository/memdb/memdb.go
Original file line number Diff line number Diff line change
@@ -1,36 +1,86 @@
package memdb

import (
"coupon_service/internal/service/entity"
"errors"
"fmt"
)

type Config struct{}
"sync"

type repository interface {
FindByCode(string) (*entity.Coupon, error)
Save(entity.Coupon) error
}
"coupon_service/internal/service/entity"
)

// Repository defines the in-memory storage for Coupons.
// It implements the repository interface.
type Repository struct {
entries map[string]entity.Coupon
entries map[string]*entity.Coupon
mu sync.RWMutex
}

func New() *Repository {
// NewRepository creates and returns a new Repository instance.
func NewRepository() *Repository {
return &Repository{
entries: make(map[string]entity.Coupon),
entries: make(map[string]*entity.Coupon),
}
}

// RepositoryInterface defines the methods that the Repository implements.
// Exported for external usage if needed.
type RepositoryInterface interface {
FindByCode(string) (*entity.Coupon, error)
Save(*entity.Coupon) error
Delete(string) error
}

// Custom errors for better error handling.
var (
ErrCouponNotFound = errors.New("coupon not found")
ErrInvalidCoupon = errors.New("invalid coupon")
)

// FindByCode retrieves a Coupon by its code.
// It returns a copy of the Coupon to prevent external modifications.
func (r *Repository) FindByCode(code string) (*entity.Coupon, error) {
coupon, ok := r.entries[code]
if !ok {
return nil, fmt.Errorf("Coupon not found")
r.mu.RLock()
defer r.mu.RUnlock()

coupon, exists := r.entries[code]
if !exists {
return nil, ErrCouponNotFound
}

// Return a copy to maintain immutability.
couponCopy := *coupon
return &couponCopy, nil
}

// Save stores a Coupon in the repository.
// It returns an error if the coupon is nil or has an empty code.
func (r *Repository) Save(coupon *entity.Coupon) error {
if coupon == nil {
return ErrInvalidCoupon
}
if coupon.Code == "" {
return fmt.Errorf("%w: coupon code is empty", ErrInvalidCoupon)
}
return &coupon, nil

r.mu.Lock()
defer r.mu.Unlock()

// Store a copy to prevent external modifications affecting the repository.
couponCopy := *coupon
r.entries[coupon.Code] = &couponCopy
return nil
}

func (r *Repository) Save(coupon entity.Coupon) error {
r.entries[coupon.Code] = coupon
// Delete removes a Coupon from the repository by its code.
// It returns an error if the coupon does not exist.
func (r *Repository) Delete(code string) error {
r.mu.Lock()
defer r.mu.Unlock()

if _, exists := r.entries[code]; !exists {
return ErrCouponNotFound
}

delete(r.entries, code)
return nil
}
160 changes: 160 additions & 0 deletions internal/repository/memdb/memdb_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,160 @@
package memdb

import (
"fmt"
"sync"
"testing"

"coupon_service/internal/service/entity"

"github.com/stretchr/testify/assert"
)

func TestRepository_Save(t *testing.T) {
repo := NewRepository()

t.Run("Save and retrieve a valid coupon", func(t *testing.T) {
coupon := &entity.Coupon{
ID: "1",
Code: "DISCOUNT10",
Discount: 10,
MinBasketValue: 50,
}

err := repo.Save(coupon)
assert.NoError(t, err, "Expected no error when saving a valid coupon")

storedCoupon, err := repo.FindByCode("DISCOUNT10")
assert.NoError(t, err, "Expected to find the saved coupon")
assert.Equal(t, coupon, storedCoupon, "Stored coupon should match the saved coupon")
})

t.Run("Overwrite an existing coupon", func(t *testing.T) {
coupon1 := &entity.Coupon{
ID: "2",
Code: "SAVE20",
Discount: 20,
MinBasketValue: 100,
}

coupon2 := &entity.Coupon{
ID: "3",
Code: "SAVE20",
Discount: 25,
MinBasketValue: 150,
}

err := repo.Save(coupon1)
assert.NoError(t, err, "Expected no error when saving the first coupon")

err = repo.Save(coupon2)
assert.NoError(t, err, "Expected no error when overwriting the existing coupon")

storedCoupon, err := repo.FindByCode("SAVE20")
assert.NoError(t, err, "Expected to find the overwritten coupon")
assert.Equal(t, coupon2, storedCoupon, "Stored coupon should match the latest saved coupon")
})
}

func TestRepository_FindByCode(t *testing.T) {
repo := NewRepository()

t.Run("Find existing coupon", func(t *testing.T) {
coupon := &entity.Coupon{
ID: "4",
Code: "SUMMER20",
Discount: 20,
MinBasketValue: 100,
}

err := repo.Save(coupon)
assert.NoError(t, err, "Expected no error when saving a valid coupon")

retrievedCoupon, err := repo.FindByCode("SUMMER20")
assert.NoError(t, err, "Expected to find the existing coupon")
assert.Equal(t, coupon, retrievedCoupon, "Retrieved coupon should match the saved coupon")
})

t.Run("Attempt to find a non-existent coupon", func(t *testing.T) {
_, err := repo.FindByCode("NONEXISTENT")
assert.Error(t, err, "Expected an error when finding a non-existent coupon")
assert.Equal(t, ErrCouponNotFound, err, "Error should be ErrCouponNotFound")
})
}

func TestRepository_Delete(t *testing.T) {
repo := NewRepository()

t.Run("Delete existing coupon", func(t *testing.T) {
coupon := &entity.Coupon{
ID: "5",
Code: "WINTER30",
Discount: 30,
MinBasketValue: 150,
}

err := repo.Save(coupon)
assert.NoError(t, err, "Expected no error when saving a valid coupon")

err = repo.Delete("WINTER30")
assert.NoError(t, err, "Expected no error when deleting an existing coupon")

// Verify that the coupon is deleted
_, err = repo.FindByCode("WINTER30")
assert.Error(t, err, "Expected an error when finding a deleted coupon")
assert.Equal(t, ErrCouponNotFound, err, "Error should be ErrCouponNotFound")
})

t.Run("Attempt to delete a non-existent coupon", func(t *testing.T) {
err := repo.Delete("NONEXISTENT")
assert.Error(t, err, "Expected an error when deleting a non-existent coupon")
assert.Equal(t, ErrCouponNotFound, err, "Error should be ErrCouponNotFound")
})
}

func TestRepository_Concurrency(t *testing.T) {
repo := NewRepository()
var wg sync.WaitGroup
numGoroutines := 50
couponCodePrefix := "CONCUR_"

saveCoupons := func(start, end int) {
defer wg.Done()
for i := start; i < end; i++ {
coupon := &entity.Coupon{
ID: fmt.Sprintf("%d", i),
Code: fmt.Sprintf("%s%d", couponCodePrefix, i),
Discount: i % 100,
MinBasketValue: i * 10,
}
repo.Save(coupon)
}
}

findCoupons := func(start, end int) {
defer wg.Done()
for i := start; i < end; i++ {
repo.FindByCode(fmt.Sprintf("%s%d", couponCodePrefix, i))
}
}

for i := 0; i < numGoroutines; i++ {
wg.Add(1)
go saveCoupons(i*100, (i+1)*100)
}

for i := 0; i < numGoroutines; i++ {
wg.Add(1)
go findCoupons(i*100, (i+1)*100)
}

wg.Wait()

for i := 0; i < numGoroutines*100; i += 1000 {
code := fmt.Sprintf("%s%d", couponCodePrefix, i)
coupon, err := repo.FindByCode(code)
if err == nil {
assert.Equal(t, code, coupon.Code, "Coupon code should match")
}
}
}
4 changes: 2 additions & 2 deletions internal/service/service.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import (

type Repository interface {
FindByCode(string) (*Coupon, error)
Save(Coupon) error
Save(*Coupon) error
}

type Service struct {
Expand Down Expand Up @@ -48,7 +48,7 @@ func (s Service) CreateCoupon(discount int, code string, minBasketValue int) (st
ID: uuid.NewString(),
}

if err := s.repo.Save(coupon); err != nil {
if err := s.repo.Save(&coupon); err != nil {
return "", err
}
return coupon.ID, nil
Expand Down
21 changes: 21 additions & 0 deletions release.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
[Metadata]

Author = "Matheus Politano"
Release_file_version = "2.1"

[Description]

Notes = """
Improve memdb to prevent future issues
"""



[[Digest.Improvements]]
Name = "8-improve-memdb"
Issue = 8
Description = """
Add extra methods, improve the Concurrency, add extra Validation and Immutability
"""