Skip to content

Commit df81491

Browse files
authored
Merge pull request ReactiveCocoa#24 from Lightricks/feature/switch-to-latest-dispose-inner-signal-first
RACSignal+Operations: rewrite -switchToLatest without -takeUntil:.
2 parents 5406a55 + 8d6619b commit df81491

File tree

2 files changed

+112
-17
lines changed

2 files changed

+112
-17
lines changed

ReactiveObjC/RACSignal+Operations.m

+62-12
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
#import "RACUnit.h"
2929
#import <libkern/OSAtomic.h>
3030
#import <objc/runtime.h>
31+
#import <os/lock.h>
3132

3233
NSString * const RACSignalErrorDomain = @"RACSignalErrorDomain";
3334

@@ -852,22 +853,71 @@ - (RACSignal *)takeUntilReplacement:(RACSignal *)replacement {
852853

853854
- (RACSignal *)switchToLatest {
854855
return [[RACSignal createSignal:^(id<RACSubscriber> subscriber) {
855-
RACMulticastConnection *connection = [self publish];
856+
__block BOOL outerCompleted = NO;
857+
__block BOOL innerCompleted = NO;
858+
__block NSUInteger currentInnerIndex = 0;
859+
860+
RACSerialDisposable *innerDisposable = [[RACSerialDisposable alloc] init];
861+
__block os_unfair_lock lock = OS_UNFAIR_LOCK_INIT;
862+
863+
RACDisposable *outerDisposable = [self subscribeNext:^(RACSignal *signal) {
864+
NSCAssert(!signal || [signal isKindOfClass:RACSignal.class], @"-switchToLatest expects signals as values, instead we got: %@", signal);
865+
signal = signal ?: [RACSignal empty];
866+
867+
// Ensure no further data is sent from the previous inner signal.
868+
[innerDisposable.disposable dispose];
869+
870+
os_unfair_lock_lock(&lock);
871+
NSUInteger innerIndex = ++currentInnerIndex;
872+
// We got a new inner signal, so reset the inner completion flag only if the outer signal has
873+
// not completed already, otherwise we may miss sending a completion in a different code path.
874+
innerCompleted &= outerCompleted;
875+
BOOL alreadyCompleted = innerCompleted;
876+
os_unfair_lock_unlock(&lock);
877+
878+
// Avoid subscribing to a new inner signal which may send more values if the signal already
879+
// completed. In this case we're already going to send a completion via the inner or the outer
880+
// signals in a different code path.
881+
if (alreadyCompleted) {
882+
return;
883+
}
856884

857-
RACDisposable *subscriptionDisposable = [[connection.signal
858-
flattenMap:^(RACSignal *x) {
859-
NSCAssert(x == nil || [x isKindOfClass:RACSignal.class], @"-switchToLatest requires that the source signal (%@) send signals. Instead we got: %@", self, x);
885+
// Required to guarantee that the previous inner signal will not send any new values from now
886+
// on.
887+
[innerDisposable.disposable dispose];
860888

861-
// -concat:[RACSignal never] prevents completion of the receiver from
862-
// prematurely terminating the inner signal.
863-
return [x takeUntil:[connection.signal concat:[RACSignal never]]];
864-
}]
865-
subscribe:subscriber];
889+
innerDisposable.disposable = [signal subscribeNext:^(id x) {
890+
[subscriber sendNext:x];
891+
} error:^(NSError *error) {
892+
[subscriber sendError:error];
893+
} completed:^{
894+
os_unfair_lock_lock(&lock);
895+
if (innerIndex == currentInnerIndex) {
896+
innerCompleted = YES;
897+
}
898+
BOOL sendCompleted = innerCompleted && outerCompleted;
899+
os_unfair_lock_unlock(&lock);
900+
901+
if (sendCompleted) {
902+
[subscriber sendCompleted];
903+
}
904+
}];
905+
} error:^(NSError *error) {
906+
[subscriber sendError:error];
907+
} completed:^{
908+
os_unfair_lock_lock(&lock);
909+
BOOL localInnerCompleted = innerCompleted;
910+
outerCompleted = YES;
911+
os_unfair_lock_unlock(&lock);
912+
913+
if (localInnerCompleted) {
914+
[subscriber sendCompleted];
915+
}
916+
}];
866917

867-
RACDisposable *connectionDisposable = [connection connect];
868918
return [RACDisposable disposableWithBlock:^{
869-
[subscriptionDisposable dispose];
870-
[connectionDisposable dispose];
919+
[innerDisposable dispose];
920+
[outerDisposable dispose];
871921
}];
872922
}] setNameWithFormat:@"[%@] -switchToLatest", self.name];
873923
}

ReactiveObjCTests/RACSignalSpec.m

+50-5
Original file line numberDiff line numberDiff line change
@@ -252,7 +252,7 @@ + (void)configure:(Configuration *)configuration {
252252
expect(@(secondSubscribed)).to(beFalsy());
253253
expect(@(errored)).to(beTruthy());
254254
});
255-
255+
256256
qck_it(@"should not retain signals that are subscribed", ^{
257257
__weak RACSignal *weakSignal;
258258
@autoreleasepool {
@@ -1985,6 +1985,20 @@ + (void)configure:(Configuration *)configuration {
19851985
expect(@(completed)).to(beTruthy());
19861986
});
19871987

1988+
qck_it(@"should dispose previous inner signal before subscribing to new inner signal", ^{
1989+
RACSubject *otherSignal = [RACSubject subject];
1990+
RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
1991+
[otherSignal sendNext:@"foo"];
1992+
return nil;
1993+
}];
1994+
1995+
[subject sendNext:otherSignal];
1996+
expect(values).to(equal(@[]));
1997+
1998+
[subject sendNext:signal];
1999+
expect(values).to(equal(@[]));
2000+
});
2001+
19882002
qck_it(@"should accept nil signals", ^{
19892003
[subject sendNext:nil];
19902004
[subject sendNext:[RACSignal createSignal:^ RACDisposable * (id<RACSubscriber> subscriber) {
@@ -1997,6 +2011,37 @@ + (void)configure:(Configuration *)configuration {
19972011
expect(values).to(equal(expected));
19982012
});
19992013

2014+
qck_it(@"should deliver in right order when inner signals deliver on multiple schedulers", ^{
2015+
for (int i = 0; i < 50; ++i) {
2016+
@autoreleasepool {
2017+
RACSignal *signalA = [[RACSignal return:@1]
2018+
subscribeOn:[RACScheduler scheduler]];
2019+
2020+
RACSignal *signalB = [[RACSignal return:@2]
2021+
subscribeOn:[RACScheduler scheduler]];
2022+
2023+
__block NSMutableArray *values = [NSMutableArray array];
2024+
RACSubject *subject = [RACSubject subject];
2025+
2026+
__block atomic_bool completed = NO;
2027+
[[subject
2028+
switchToLatest]
2029+
subscribeNext:^(id x) {
2030+
[values addObject:x];
2031+
} completed:^{
2032+
expect(values.lastObject).to(equal(@2));
2033+
completed = YES;
2034+
}];
2035+
2036+
[subject sendNext:signalA];
2037+
[subject sendNext:signalB];
2038+
[subject sendCompleted];
2039+
2040+
expect(completed).toEventually(beTruthy());
2041+
}
2042+
}
2043+
});
2044+
20002045
qck_it(@"should return a cold signal", ^{
20012046
__block NSUInteger subscriptions = 0;
20022047
RACSignal *signalOfSignals = [RACSignal createSignal:^ RACDisposable * (id<RACSubscriber> subscriber) {
@@ -3264,7 +3309,7 @@ + (void)configure:(Configuration *)configuration {
32643309
qck_beforeEach(^{
32653310
signal = [[[RACSignal return:@0] concat:[RACSignal return:@1]] concat:[RACSignal return:@2]];
32663311
});
3267-
3312+
32683313
qck_it(@"should return true when the predicate is truthy for at least one value", ^{
32693314
RACSignal *any = [signal any:^BOOL(NSNumber *value) {
32703315
return value.integerValue > 0;
@@ -3280,7 +3325,7 @@ + (void)configure:(Configuration *)configuration {
32803325
expect(values).to(equal(@[@YES]));
32813326
expect(@(completed)).to(equal(@YES));
32823327
});
3283-
3328+
32843329
qck_it(@"should return false when the predicate is falsy for all values", ^{
32853330
RACSignal *any = [signal any:^BOOL(NSNumber *value) {
32863331
return value.integerValue == 3;
@@ -3322,7 +3367,7 @@ + (void)configure:(Configuration *)configuration {
33223367
qck_beforeEach(^{
33233368
signal = [[[RACSignal return:@0] concat:[RACSignal return:@1]] concat:[RACSignal return:@2]];
33243369
});
3325-
3370+
33263371
qck_it(@"should return true when all values pass", ^{
33273372
RACSignal *all = [signal all:^BOOL(NSNumber *value) {
33283373
return value.integerValue >= 0;
@@ -3338,7 +3383,7 @@ + (void)configure:(Configuration *)configuration {
33383383
expect(values).to(equal(@[@YES]));
33393384
expect(@(completed)).to(equal(@YES));
33403385
});
3341-
3386+
33423387
qck_it(@"should return false when at least one value fails", ^{
33433388
RACSignal *all = [signal all:^BOOL(NSNumber *value) {
33443389
return value.integerValue < 2;

0 commit comments

Comments
 (0)