Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
B
brpc
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
brpc
Commits
aaf20ce8
Commit
aaf20ce8
authored
Oct 23, 2018
by
TousakaRin
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Make the interface more robust
parent
74929cfc
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
38 additions
and
22 deletions
+38
-22
circuit_breaker.cpp
src/brpc/circuit_breaker.cpp
+15
-5
circuit_breaker.h
src/brpc/circuit_breaker.h
+7
-2
brpc_circuit_breaker_unittest.cpp
test/brpc_circuit_breaker_unittest.cpp
+16
-15
No files found.
src/brpc/circuit_breaker.cpp
View file @
aaf20ce8
...
@@ -160,12 +160,20 @@ CircuitBreaker::CircuitBreaker()
...
@@ -160,12 +160,20 @@ CircuitBreaker::CircuitBreaker()
FLAGS_circuit_breaker_short_window_error_percent
)
FLAGS_circuit_breaker_short_window_error_percent
)
,
_last_reset_time_ms
(
butil
::
cpuwide_time_ms
())
,
_last_reset_time_ms
(
butil
::
cpuwide_time_ms
())
,
_isolation_duration_ms
(
FLAGS_circuit_breaker_min_isolation_duration_ms
)
,
_isolation_duration_ms
(
FLAGS_circuit_breaker_min_isolation_duration_ms
)
,
_isolated_times
(
0
)
{
,
_isolated_times
(
0
)
,
_broken
(
false
)
{
}
}
bool
CircuitBreaker
::
OnCallEnd
(
int
error_code
,
int64_t
latency
)
{
bool
CircuitBreaker
::
OnCallEnd
(
int
error_code
,
int64_t
latency
)
{
return
_long_window
.
OnCallEnd
(
error_code
,
latency
)
&&
if
(
_broken
.
load
(
butil
::
memory_order_relaxed
))
{
_short_window
.
OnCallEnd
(
error_code
,
latency
);
return
false
;
}
if
(
_long_window
.
OnCallEnd
(
error_code
,
latency
)
&&
_short_window
.
OnCallEnd
(
error_code
,
latency
))
{
return
true
;
}
MarkAsBroken
();
return
false
;
}
}
void
CircuitBreaker
::
Reset
()
{
void
CircuitBreaker
::
Reset
()
{
...
@@ -175,8 +183,10 @@ void CircuitBreaker::Reset() {
...
@@ -175,8 +183,10 @@ void CircuitBreaker::Reset() {
}
}
void
CircuitBreaker
::
MarkAsBroken
()
{
void
CircuitBreaker
::
MarkAsBroken
()
{
++
_isolated_times
;
if
(
_broken
.
exchange
(
true
,
butil
::
memory_order_acquire
))
{
UpdateIsolationDuration
();
++
_isolated_times
;
UpdateIsolationDuration
();
}
}
}
int
CircuitBreaker
::
health_score
()
const
{
int
CircuitBreaker
::
health_score
()
const
{
...
...
src/brpc/circuit_breaker.h
View file @
aaf20ce8
...
@@ -31,6 +31,9 @@ public:
...
@@ -31,6 +31,9 @@ public:
// be isolated. Otherwise return true.
// be isolated. Otherwise return true.
// error_code: Error_code of this call, 0 means success.
// error_code: Error_code of this call, 0 means success.
// latency: Time cost of this call.
// latency: Time cost of this call.
// Note: Once OnCallEnd() determined that a node needs to be isolated,
// it will always return false until you call Reset(). Usually Reset()
// will be called in the health check thread.
bool
OnCallEnd
(
int
error_code
,
int64_t
latency
);
bool
OnCallEnd
(
int
error_code
,
int64_t
latency
);
// Reset CircuitBreaker and clear history data. will erase the historical
// Reset CircuitBreaker and clear history data. will erase the historical
...
@@ -38,8 +41,9 @@ public:
...
@@ -38,8 +41,9 @@ public:
// ensure that no one else is calling OnCallEnd.
// ensure that no one else is calling OnCallEnd.
void
Reset
();
void
Reset
();
// Mark the Socket as broken. You should NOT call this method multiple
// Mark the Socket as broken. Call this method when you want to isolate a
// times in succession.
// node in advance. When this method is called multiple times in succession,
// only the first call will take effect.
void
MarkAsBroken
();
void
MarkAsBroken
();
// The closer to 100, the less recent errors occurred, and 0 means that
// The closer to 100, the less recent errors occurred, and 0 means that
...
@@ -87,6 +91,7 @@ private:
...
@@ -87,6 +91,7 @@ private:
int64_t
_last_reset_time_ms
;
int64_t
_last_reset_time_ms
;
int
_isolation_duration_ms
;
int
_isolation_duration_ms
;
int
_isolated_times
;
int
_isolated_times
;
butil
::
atomic
<
bool
>
_broken
;
};
};
}
// namespace brpc
}
// namespace brpc
...
...
test/brpc_circuit_breaker_unittest.cpp
View file @
aaf20ce8
...
@@ -60,10 +60,14 @@ struct FeedbackControl {
...
@@ -60,10 +60,14 @@ struct FeedbackControl {
:
_req_num
(
req_num
)
:
_req_num
(
req_num
)
,
_error_percent
(
error_percent
)
,
_error_percent
(
error_percent
)
,
_circuit_breaker
(
circuit_breaker
)
,
_circuit_breaker
(
circuit_breaker
)
,
_healthy_cnt
(
0
)
,
_unhealthy_cnt
(
0
)
,
_healthy
(
true
)
{}
,
_healthy
(
true
)
{}
int
_req_num
;
int
_req_num
;
int
_error_percent
;
int
_error_percent
;
brpc
::
CircuitBreaker
*
_circuit_breaker
;
brpc
::
CircuitBreaker
*
_circuit_breaker
;
int
_healthy_cnt
;
int
_unhealthy_cnt
;
bool
_healthy
;
bool
_healthy
;
};
};
...
@@ -87,8 +91,10 @@ protected:
...
@@ -87,8 +91,10 @@ protected:
healthy
=
fc
->
_circuit_breaker
->
OnCallEnd
(
kErrorCodeForSucc
,
kLatency
);
healthy
=
fc
->
_circuit_breaker
->
OnCallEnd
(
kErrorCodeForSucc
,
kLatency
);
}
}
fc
->
_healthy
=
healthy
;
fc
->
_healthy
=
healthy
;
if
(
!
healthy
)
{
if
(
healthy
)
{
break
;
++
fc
->
_healthy_cnt
;
}
else
{
++
fc
->
_unhealthy_cnt
;
}
}
}
}
return
fc
;
return
fc
;
...
@@ -120,11 +126,9 @@ TEST_F(CircuitBreakerTest, should_not_isolate) {
...
@@ -120,11 +126,9 @@ TEST_F(CircuitBreakerTest, should_not_isolate) {
void
*
ret_data
=
NULL
;
void
*
ret_data
=
NULL
;
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
EXPECT_EQ
(
fc
->
_unhealthy_cnt
,
0
);
EXPECT_TRUE
(
fc
->
_healthy
);
EXPECT_TRUE
(
fc
->
_healthy
);
}
}
// MarkAsBroken shoul be called only once.
_circuit_breaker
.
MarkAsBroken
();
_circuit_breaker
.
Reset
();
}
}
TEST_F
(
CircuitBreakerTest
,
should_isolate
)
{
TEST_F
(
CircuitBreakerTest
,
should_isolate
)
{
...
@@ -135,15 +139,12 @@ TEST_F(CircuitBreakerTest, should_isolate) {
...
@@ -135,15 +139,12 @@ TEST_F(CircuitBreakerTest, should_isolate) {
void
*
ret_data
=
NULL
;
void
*
ret_data
=
NULL
;
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
EXPECT_GT
(
fc
->
_unhealthy_cnt
,
0
);
EXPECT_FALSE
(
fc
->
_healthy
);
EXPECT_FALSE
(
fc
->
_healthy
);
}
}
// MarkAsBroken shoul be called only once.
_circuit_breaker
.
MarkAsBroken
();
_circuit_breaker
.
Reset
();
}
}
TEST_F
(
CircuitBreakerTest
,
isolation_duration_grow
)
{
TEST_F
(
CircuitBreakerTest
,
isolation_duration_grow
)
{
EXPECT_EQ
(
_circuit_breaker
.
isolation_duration_ms
(),
kMinIsolationDurationMs
);
_circuit_breaker
.
Reset
();
_circuit_breaker
.
Reset
();
std
::
vector
<
pthread_t
>
thread_list
;
std
::
vector
<
pthread_t
>
thread_list
;
std
::
vector
<
std
::
unique_ptr
<
FeedbackControl
>>
fc_list
;
std
::
vector
<
std
::
unique_ptr
<
FeedbackControl
>>
fc_list
;
...
@@ -153,9 +154,9 @@ TEST_F(CircuitBreakerTest, isolation_duration_grow) {
...
@@ -153,9 +154,9 @@ TEST_F(CircuitBreakerTest, isolation_duration_grow) {
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
EXPECT_FALSE
(
fc
->
_healthy
);
EXPECT_FALSE
(
fc
->
_healthy
);
EXPECT_LE
(
fc
->
_healthy_cnt
,
kShortWindowSize
);
EXPECT_GT
(
fc
->
_unhealthy_cnt
,
0
);
}
}
// MarkAsBroken shoul be called only once.
_circuit_breaker
.
MarkAsBroken
();
EXPECT_EQ
(
_circuit_breaker
.
isolation_duration_ms
(),
kMinIsolationDurationMs
*
2
);
EXPECT_EQ
(
_circuit_breaker
.
isolation_duration_ms
(),
kMinIsolationDurationMs
*
2
);
_circuit_breaker
.
Reset
();
_circuit_breaker
.
Reset
();
...
@@ -166,9 +167,9 @@ TEST_F(CircuitBreakerTest, isolation_duration_grow) {
...
@@ -166,9 +167,9 @@ TEST_F(CircuitBreakerTest, isolation_duration_grow) {
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
EXPECT_FALSE
(
fc
->
_healthy
);
EXPECT_FALSE
(
fc
->
_healthy
);
EXPECT_LE
(
fc
->
_healthy_cnt
,
kShortWindowSize
);
EXPECT_GT
(
fc
->
_unhealthy_cnt
,
0
);
}
}
// MarkAsBroken shoul be called only once.
_circuit_breaker
.
MarkAsBroken
();
EXPECT_EQ
(
_circuit_breaker
.
isolation_duration_ms
(),
kMinIsolationDurationMs
*
4
);
EXPECT_EQ
(
_circuit_breaker
.
isolation_duration_ms
(),
kMinIsolationDurationMs
*
4
);
_circuit_breaker
.
Reset
();
_circuit_breaker
.
Reset
();
...
@@ -179,8 +180,8 @@ TEST_F(CircuitBreakerTest, isolation_duration_grow) {
...
@@ -179,8 +180,8 @@ TEST_F(CircuitBreakerTest, isolation_duration_grow) {
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
EXPECT_EQ
(
pthread_join
(
thread_list
[
i
],
&
ret_data
),
0
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
FeedbackControl
*
fc
=
static_cast
<
FeedbackControl
*>
(
ret_data
);
EXPECT_FALSE
(
fc
->
_healthy
);
EXPECT_FALSE
(
fc
->
_healthy
);
EXPECT_LE
(
fc
->
_healthy_cnt
,
kShortWindowSize
);
EXPECT_GT
(
fc
->
_unhealthy_cnt
,
0
);
}
}
// MarkAsBroken shoul be called only once.
_circuit_breaker
.
MarkAsBroken
();
EXPECT_EQ
(
_circuit_breaker
.
isolation_duration_ms
(),
kMinIsolationDurationMs
);
EXPECT_EQ
(
_circuit_breaker
.
isolation_duration_ms
(),
kMinIsolationDurationMs
);
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment