从 doNext 块发送错误?
Send error from doNext block?
我正在使用 ReactiveCocoa 和 Overcoat/Mantle/AFNetworking 从 RESTful API.
获取数据并对用户进行身份验证
这是管理登录按钮和凭据文本字段的登录视图控制器中的代码:
@weakify(self);
self.loginButton.rac_command =
[[RACCommand alloc] initWithEnabled:validCredentials
signalBlock:^RACSignal *(id input) {
@strongify(self);
return [[PFUserManager sharedManager] logInUser:self.usernameTextField.text
password:self.passwordTextField.text];
}];
// Handle errors for the login command
[self.loginButton.rac_command.errors subscribeNext:^(NSError *error) {
// Present the error message
[PFErrorAlertFactory showOVCError:error];
}];
// Take care of the signal from the request
[[self.loginButton.rac_command.executionSignals flatten] subscribeNext:^(NSNumber *success) {
@strongify(self);
[self clearTextFields];
[self.flowController controllerForMainScreen]; // Transition to "logged in state"
} error:^(NSError *error) {
@strongify(self);
[self clearTextFields];
}];
我在单例 UserManager 上有这个方法 class:
- (RACSignal *)logInUser:(NSString *)username password:(NSString *)password {
// Return a cold signal that sends next and complete when user is authenticated and error if authentication failed.
PFAPIClient *client = [[PFAPIClient alloc] initWithUsername:username password:password];
@weakify(self);
RACSignal *loginSignal = [[client rac_POST:kAuthenticationResourcePath parameters:nil] doNext:^(OVCResponse *response) {
@strongify(self);
self.currentUser = response.result;
NSError *error;
[SSKeychain setPassword:password forService:kKeychainServiceKey account:self.currentUser.username error:&error];
if (error) {
[PFErrorAlertFactory showLocalizedDescriptionOfError:error];
}
}];
return loginSignal;
}
一切都很好,将此信号用作按钮的 RACCommand 信号。我在登录视图控制器中处理下一个、错误和完成的事件,它工作正常。
正如您在 UserManager 代码中看到的那样,在 doNext 块中,如果 Keychain 方法 returns 一个,我会显示一个错误。我有点不确定这个错误处理是否属于这个 class.
它确实有效,错误显示为 UIAlertView,但是这个 UserManager class 真的应该负责显示错误吗?
来自 rac_POST 信号的错误由登录视图控制器处理,我也想在这里处理来自 Keychain 方法的错误。是否可以从 doNext 块中向 rac_POST 信号的订阅者发送错误?虽然我缺少指向订阅者的指针......同样,如果 Keychain 方法中发生错误,信号仍会发送下一个并完成,并且就调用视图控制器所知,登录成功。这显然不是它应该的工作方式。
有没有其他首选方法来处理整个情况?我知道 doNext 块中的副作用不是首选,但在这种情况下我看不到其他解决方案,因为我希望 UserManager 拥有此方法并能够设置自己的 currentUser。我是否应该将其包装在一个新信号中并显式发送下一个、完成和错误?
此致,
詹斯
您可以创建 SSKeychain
类别:
@interface SSKeychain (RACExtension)
- (RACSignal*)rac_setPassword:(NSString *)password forService:(NSString *)serviceName account:(NSString *)account;
@end
@implementation SSKeychain (RACExtension)
- (RACSignal*)rac_setPassword:(NSString *)password forService:(NSString *)serviceName account:(NSString *)account
{
[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSError *error;
BOOL result = [SSKeychain setPassword:password forService:service account:account error:&error];
if (result) {
[subscriber sendNext:@(result)];
[subscriber sendCompleted];
} else {
[subscriber sendError:error];
}
}];
}
@end
然后用你的 UserManager
:
- (RACSignal *)logInUser:(NSString *)username password:(NSString *)password
{
PFAPIClient *client = [[PFAPIClient alloc] initWithUsername:username password:password];
@weakify(self);
return [[[client rac_POST:kAuthenticationResourcePath parameters:nil]
flattenMap:^RACStream *(OVCResponse *response){
@strongify(self)
return [[SSKeychain rac_setPassword:password forService:kKeychainServiceKey account:self.currentUser.username]
mapReplace:response];
}]
doNext:^(OVCResponse *response){
@strongify(self)
self.currentUser = response.result;
}]
}
我正在使用 ReactiveCocoa 和 Overcoat/Mantle/AFNetworking 从 RESTful API.
获取数据并对用户进行身份验证这是管理登录按钮和凭据文本字段的登录视图控制器中的代码:
@weakify(self);
self.loginButton.rac_command =
[[RACCommand alloc] initWithEnabled:validCredentials
signalBlock:^RACSignal *(id input) {
@strongify(self);
return [[PFUserManager sharedManager] logInUser:self.usernameTextField.text
password:self.passwordTextField.text];
}];
// Handle errors for the login command
[self.loginButton.rac_command.errors subscribeNext:^(NSError *error) {
// Present the error message
[PFErrorAlertFactory showOVCError:error];
}];
// Take care of the signal from the request
[[self.loginButton.rac_command.executionSignals flatten] subscribeNext:^(NSNumber *success) {
@strongify(self);
[self clearTextFields];
[self.flowController controllerForMainScreen]; // Transition to "logged in state"
} error:^(NSError *error) {
@strongify(self);
[self clearTextFields];
}];
我在单例 UserManager 上有这个方法 class:
- (RACSignal *)logInUser:(NSString *)username password:(NSString *)password {
// Return a cold signal that sends next and complete when user is authenticated and error if authentication failed.
PFAPIClient *client = [[PFAPIClient alloc] initWithUsername:username password:password];
@weakify(self);
RACSignal *loginSignal = [[client rac_POST:kAuthenticationResourcePath parameters:nil] doNext:^(OVCResponse *response) {
@strongify(self);
self.currentUser = response.result;
NSError *error;
[SSKeychain setPassword:password forService:kKeychainServiceKey account:self.currentUser.username error:&error];
if (error) {
[PFErrorAlertFactory showLocalizedDescriptionOfError:error];
}
}];
return loginSignal;
}
一切都很好,将此信号用作按钮的 RACCommand 信号。我在登录视图控制器中处理下一个、错误和完成的事件,它工作正常。
正如您在 UserManager 代码中看到的那样,在 doNext 块中,如果 Keychain 方法 returns 一个,我会显示一个错误。我有点不确定这个错误处理是否属于这个 class.
它确实有效,错误显示为 UIAlertView,但是这个 UserManager class 真的应该负责显示错误吗?
来自 rac_POST 信号的错误由登录视图控制器处理,我也想在这里处理来自 Keychain 方法的错误。是否可以从 doNext 块中向 rac_POST 信号的订阅者发送错误?虽然我缺少指向订阅者的指针......同样,如果 Keychain 方法中发生错误,信号仍会发送下一个并完成,并且就调用视图控制器所知,登录成功。这显然不是它应该的工作方式。
有没有其他首选方法来处理整个情况?我知道 doNext 块中的副作用不是首选,但在这种情况下我看不到其他解决方案,因为我希望 UserManager 拥有此方法并能够设置自己的 currentUser。我是否应该将其包装在一个新信号中并显式发送下一个、完成和错误?
此致, 詹斯
您可以创建 SSKeychain
类别:
@interface SSKeychain (RACExtension)
- (RACSignal*)rac_setPassword:(NSString *)password forService:(NSString *)serviceName account:(NSString *)account;
@end
@implementation SSKeychain (RACExtension)
- (RACSignal*)rac_setPassword:(NSString *)password forService:(NSString *)serviceName account:(NSString *)account
{
[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSError *error;
BOOL result = [SSKeychain setPassword:password forService:service account:account error:&error];
if (result) {
[subscriber sendNext:@(result)];
[subscriber sendCompleted];
} else {
[subscriber sendError:error];
}
}];
}
@end
然后用你的 UserManager
:
- (RACSignal *)logInUser:(NSString *)username password:(NSString *)password
{
PFAPIClient *client = [[PFAPIClient alloc] initWithUsername:username password:password];
@weakify(self);
return [[[client rac_POST:kAuthenticationResourcePath parameters:nil]
flattenMap:^RACStream *(OVCResponse *response){
@strongify(self)
return [[SSKeychain rac_setPassword:password forService:kKeychainServiceKey account:self.currentUser.username]
mapReplace:response];
}]
doNext:^(OVCResponse *response){
@strongify(self)
self.currentUser = response.result;
}]
}