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

Implement circuit breaker #129

Open
wants to merge 7 commits into
base: master
Choose a base branch
from

Conversation

MitulShah1
Copy link

This PR contains Circuit Breaker Middleware

@aldas @jrhrmsll @lammel Would appreciate it if you can review this PR.

@MitulShah1
Copy link
Author

MitulShah1 commented Mar 28, 2025

@aldas I've made changes and make it more advance and optimize, can you please check again?

timeout time.Duration // Duration to stay open before transitioning to half-open
successThreshold int // Successes required to close circuit
openTimer *time.Timer // Timer for state transition from open to half-open
ctx context.Context // Context for cancellation
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this does not seem to be used.

cancel context.CancelFunc // Cancel function for cleanup
config Config // Configuration settings
now func() time.Time // Function for getting current time (useful for testing)
halfOpenSemaphore chan struct{} // Controls limited requests in half-open state
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why not use atomic.uint to count requests in flight? At instead of that openTimer *time.Timer just have atomic uint64 (timestamp nano) that is checked if halfstate has been expired.

or atleast move this semaphore specific logic to own struct + methods that are similar to https://pkg.go.dev/golang.org/x/sync/semaphore API - this would be easier to read.

or very least - all places where there is select { in this code. it must have comment what it does. You do not see often people doing semaphores with chans and it takes too much processing/mental power to understand what the codes does. Especially when it comes to maintaining. For example we Echo maintainers have to read this code in 6, 12, 24 months and understand what it does with minimal effort. I can not expect you to be around then to explain.

}

// GetState returns the current state of the circuit breaker
func (cb *CircuitBreaker) GetState() State {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Usually Go does not use getter naming. So State() would be more idiomatic.

func (cb *CircuitBreaker) AllowRequest() (bool, State) {
atomic.AddInt64(&cb.totalRequests, 1)

cb.mutex.RLock()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think guarding this small block with read lock makes little difference. This whole method should be behind read lock if it is necessary that limits are honored/guarded against races during extreme peaks.

func (cb *CircuitBreaker) GetStateStats() map[string]interface{} {
state := cb.GetState()

return map[string]interface{}{
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

these stats should have their own struct with field and their types. returning map of interfaces looks like something that belongs to javascript world.

}

// HealthHandler returns an Echo handler for checking circuit breaker status
func (cb *CircuitBreaker) HealthHandler() echo.HandlerFunc {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is little bit overthinking - if there is a method that returns stats when everyone can easily build their own handlers. assume that there will be people that want to have their own structure for output.

cb.mutex.Lock()
defer cb.mutex.Unlock()

if cb.state != StateOpen {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this could be if cb.state == StateOpen { return} to get rid of 1 level of indentation

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants