从 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;
             }]
}