forked from Netflix/Hystrix
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Corrected ignoreExceptions for Observable returning methods.
- error handling for observables - tests
- Loading branch information
Showing
5 changed files
with
387 additions
and
2 deletions.
There are no files selected for viewing
37 changes: 37 additions & 0 deletions
37
...m/netflix/hystrix/contrib/javanica/test/aspectj/error/ObservableErrorPropagationTest.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
/** | ||
* Copyright 2016 Netflix, Inc. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
*/ | ||
package com.netflix.hystrix.contrib.javanica.test.aspectj.error; | ||
|
||
import com.netflix.hystrix.contrib.javanica.test.common.error.BasicErrorPropagationTest; | ||
import com.netflix.hystrix.contrib.javanica.test.common.error.BasicObservableErrorPropagationTest; | ||
import org.junit.BeforeClass; | ||
|
||
/** | ||
* Created by dmgcodevil | ||
*/ | ||
public class ObservableErrorPropagationTest extends BasicObservableErrorPropagationTest { | ||
|
||
@BeforeClass | ||
public static void setUpEnv() { | ||
System.setProperty("weavingMode", "compile"); | ||
} | ||
|
||
@Override | ||
protected UserService createUserService() { | ||
return new UserService(); | ||
} | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
260 changes: 260 additions & 0 deletions
260
...tflix/hystrix/contrib/javanica/test/common/error/BasicObservableErrorPropagationTest.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,260 @@ | ||
/** | ||
* Copyright 2016 Netflix, Inc. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
*/ | ||
package com.netflix.hystrix.contrib.javanica.test.common.error; | ||
|
||
import com.netflix.hystrix.HystrixEventType; | ||
import com.netflix.hystrix.HystrixRequestLog; | ||
import com.netflix.hystrix.contrib.javanica.annotation.HystrixCommand; | ||
import com.netflix.hystrix.contrib.javanica.test.common.BasicHystrixTest; | ||
import com.netflix.hystrix.contrib.javanica.test.common.domain.User; | ||
import org.junit.Before; | ||
import org.junit.Test; | ||
import org.mockito.MockitoAnnotations; | ||
import rx.Observable; | ||
import rx.observers.TestSubscriber; | ||
|
||
import java.util.HashMap; | ||
import java.util.Map; | ||
|
||
import static com.netflix.hystrix.contrib.javanica.test.common.CommonUtils.getHystrixCommandByKey; | ||
import static org.junit.Assert.*; | ||
import static org.mockito.Mockito.*; | ||
|
||
/** | ||
* Created by dmgcodevil | ||
*/ | ||
public abstract class BasicObservableErrorPropagationTest extends BasicHystrixTest { | ||
|
||
private static final String COMMAND_KEY = "getUserById"; | ||
|
||
private static final Map<String, User> USERS; | ||
|
||
static { | ||
USERS = new HashMap<String, User>(); | ||
USERS.put("1", new User("1", "user_1")); | ||
USERS.put("2", new User("2", "user_2")); | ||
USERS.put("3", new User("3", "user_3")); | ||
} | ||
|
||
private UserService userService; | ||
|
||
@MockitoAnnotations.Mock | ||
private FailoverService failoverService; | ||
|
||
protected abstract UserService createUserService(); | ||
|
||
@Before | ||
public void setUp() throws Exception { | ||
MockitoAnnotations.initMocks(this); | ||
userService = createUserService(); | ||
userService.setFailoverService(failoverService); | ||
} | ||
|
||
@Test | ||
public void testGetUserByBadId() throws NotFoundException { | ||
try { | ||
TestSubscriber<User> testSubscriber = new TestSubscriber<User>(); | ||
|
||
String badId = ""; | ||
userService.getUserById(badId).subscribe(testSubscriber); | ||
|
||
testSubscriber.assertError(BadRequestException.class); | ||
} finally { | ||
assertEquals(1, HystrixRequestLog.getCurrentRequest().getAllExecutedCommands().size()); | ||
com.netflix.hystrix.HystrixInvokableInfo getUserCommand = getHystrixCommandByKey(COMMAND_KEY); | ||
// will not affect metrics | ||
assertFalse(getUserCommand.getExecutionEvents().contains(HystrixEventType.FAILURE)); | ||
// and will not trigger fallback logic | ||
verify(failoverService, never()).getDefUser(); | ||
} | ||
} | ||
|
||
@Test | ||
public void testGetNonExistentUser() throws NotFoundException { | ||
try { | ||
TestSubscriber<User> testSubscriber = new TestSubscriber<User>(); | ||
|
||
userService.getUserById("4").subscribe(testSubscriber); // user with id 4 doesn't exist | ||
|
||
testSubscriber.assertError(NotFoundException.class); | ||
} finally { | ||
assertEquals(1, HystrixRequestLog.getCurrentRequest().getAllExecutedCommands().size()); | ||
com.netflix.hystrix.HystrixInvokableInfo getUserCommand = getHystrixCommandByKey(COMMAND_KEY); | ||
// will not affect metrics | ||
assertFalse(getUserCommand.getExecutionEvents().contains(HystrixEventType.FAILURE)); | ||
// and will not trigger fallback logic | ||
verify(failoverService, never()).getDefUser(); | ||
} | ||
} | ||
|
||
@Test // don't expect any exceptions because fallback must be triggered | ||
public void testActivateUser() throws NotFoundException, ActivationException { | ||
try { | ||
userService.activateUser("1").toBlocking().single(); // this method always throws ActivationException | ||
} finally { | ||
assertEquals(1, HystrixRequestLog.getCurrentRequest().getAllExecutedCommands().size()); | ||
com.netflix.hystrix.HystrixInvokableInfo activateUserCommand = getHystrixCommandByKey("activateUser"); | ||
// will not affect metrics | ||
assertTrue(activateUserCommand.getExecutionEvents().contains(HystrixEventType.FAILURE)); | ||
assertTrue(activateUserCommand.getExecutionEvents().contains(HystrixEventType.FALLBACK_SUCCESS)); | ||
// and will not trigger fallback logic | ||
verify(failoverService, atLeastOnce()).activate(); | ||
} | ||
} | ||
|
||
@Test | ||
public void testBlockUser() throws NotFoundException, ActivationException, OperationException { | ||
try { | ||
TestSubscriber<Void> testSubscriber = new TestSubscriber<Void>(); | ||
|
||
userService.blockUser("1").subscribe(testSubscriber); // this method always throws ActivationException | ||
|
||
testSubscriber.assertError(Throwable.class); | ||
assertTrue(testSubscriber.getOnErrorEvents().size() == 1); | ||
assertTrue(testSubscriber.getOnErrorEvents().get(0).getCause() instanceof OperationException); | ||
} finally { | ||
assertEquals(2, HystrixRequestLog.getCurrentRequest().getAllExecutedCommands().size()); | ||
com.netflix.hystrix.HystrixInvokableInfo activateUserCommand = getHystrixCommandByKey("blockUser"); | ||
// will not affect metrics | ||
assertTrue(activateUserCommand.getExecutionEvents().contains(HystrixEventType.FAILURE)); | ||
assertTrue(activateUserCommand.getExecutionEvents().contains(HystrixEventType.FALLBACK_FAILURE)); | ||
} | ||
} | ||
|
||
@Test | ||
public void testPropagateCauseException() throws NotFoundException { | ||
TestSubscriber<Void> testSubscriber = new TestSubscriber<Void>(); | ||
|
||
userService.deleteUser("").subscribe(testSubscriber); | ||
|
||
testSubscriber.assertError(NotFoundException.class); | ||
} | ||
|
||
public static class UserService { | ||
|
||
private FailoverService failoverService; | ||
|
||
public void setFailoverService(FailoverService failoverService) { | ||
this.failoverService = failoverService; | ||
} | ||
|
||
@HystrixCommand | ||
public Observable<Void> deleteUser(String id) throws NotFoundException { | ||
return Observable.error(new NotFoundException("")); | ||
} | ||
|
||
@HystrixCommand( | ||
commandKey = COMMAND_KEY, | ||
ignoreExceptions = { | ||
BadRequestException.class, | ||
NotFoundException.class | ||
}, | ||
fallbackMethod = "fallback") | ||
public Observable<User> getUserById(String id) throws NotFoundException { | ||
validate(id); | ||
if (!USERS.containsKey(id)) { | ||
return Observable.error(new NotFoundException("user with id: " + id + " not found")); | ||
} | ||
return Observable.just(USERS.get(id)); | ||
} | ||
|
||
|
||
@HystrixCommand( | ||
ignoreExceptions = {BadRequestException.class, NotFoundException.class}, | ||
fallbackMethod = "activateFallback") | ||
public Observable<Void> activateUser(String id) throws NotFoundException, ActivationException { | ||
validate(id); | ||
if (!USERS.containsKey(id)) { | ||
return Observable.error(new NotFoundException("user with id: " + id + " not found")); | ||
} | ||
// always throw this exception | ||
return Observable.error(new ActivationException("user cannot be activate")); | ||
} | ||
|
||
@HystrixCommand( | ||
ignoreExceptions = {BadRequestException.class, NotFoundException.class}, | ||
fallbackMethod = "blockUserFallback") | ||
public Observable<Void> blockUser(String id) throws NotFoundException, OperationException { | ||
validate(id); | ||
if (!USERS.containsKey(id)) { | ||
return Observable.error(new NotFoundException("user with id: " + id + " not found")); | ||
} | ||
// always throw this exception | ||
return Observable.error(new OperationException("user cannot be blocked")); | ||
} | ||
|
||
private Observable<User> fallback(String id) { | ||
return failoverService.getDefUser(); | ||
} | ||
|
||
private Observable<Void> activateFallback(String id) { | ||
return failoverService.activate(); | ||
} | ||
|
||
@HystrixCommand(ignoreExceptions = {RuntimeException.class}) | ||
private Observable<Void> blockUserFallback(String id) { | ||
return Observable.error(new RuntimeOperationException("blockUserFallback has failed")); | ||
} | ||
|
||
private void validate(String val) throws BadRequestException { | ||
if (val == null || val.length() == 0) { | ||
throw new BadRequestException("parameter cannot be null ot empty"); | ||
} | ||
} | ||
} | ||
|
||
private class FailoverService { | ||
public Observable<User> getDefUser() { | ||
return Observable.just(new User("def", "def")); | ||
} | ||
|
||
public Observable<Void> activate() { | ||
return Observable.empty(); | ||
} | ||
} | ||
|
||
// exceptions | ||
private static class NotFoundException extends Exception { | ||
private NotFoundException(String message) { | ||
super(message); | ||
} | ||
} | ||
|
||
private static class BadRequestException extends RuntimeException { | ||
private BadRequestException(String message) { | ||
super(message); | ||
} | ||
} | ||
|
||
private static class ActivationException extends Exception { | ||
private ActivationException(String message) { | ||
super(message); | ||
} | ||
} | ||
|
||
private static class OperationException extends Throwable { | ||
private OperationException(String message) { | ||
super(message); | ||
} | ||
} | ||
|
||
private static class RuntimeOperationException extends RuntimeException { | ||
private RuntimeOperationException(String message) { | ||
super(message); | ||
} | ||
} | ||
|
||
} |
Oops, something went wrong.