Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add ability to set an object level TTL #209

Merged
merged 39 commits into from
May 11, 2018
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
a08b66c
Add new `ageLimit` setters in PINCaching protocol
mjlazar Dec 5, 2017
3d7714e
Implement `ageLimit` methods in PINDiskCache
mjlazar Dec 5, 2017
d1c796a
Implement `ageLimit` methods in PINMemoryCache
mjlazar Dec 5, 2017
f7d6d3c
Implement `ageLimit` methods in PINCache
mjlazar Dec 5, 2017
336d401
Honor `ageLimit` (at both cache and object level) in -[PINDiskCache c…
mjlazar Dec 5, 2017
6be8d22
Honor `ageLimit` (at both cache and object level) in -[PINDiskCache c…
mjlazar Dec 5, 2017
c2dfcd6
Merge branch 'object-level-ttl-override' of github.com:mjlazar/PINCac…
mjlazar Dec 5, 2017
4b05c88
Update CHANGELOG
mjlazar Dec 16, 2017
0d4e71d
Add new NSDate category for swizzling +[NSDate date] in unit tests
mjlazar Feb 16, 2018
d4d7640
Change memory and disk caches to use +[NDate date] instead of -[NSDat…
mjlazar Feb 16, 2018
b4b7ac2
Add unit tests for object-level TTLs
mjlazar Feb 16, 2018
171de47
Update header documentation regarding object-level age limits
mjlazar Feb 16, 2018
723c326
Clear previously set object age limits if setting to zero
mjlazar Feb 16, 2018
06e331c
Add assertions to ttlCache is set to YES if object-level age limits a…
mjlazar Feb 16, 2018
3895432
Update header documentation to describe behavior of overridden object…
mjlazar Feb 16, 2018
a4466b6
Remove `hasAgeLimit` logic
mjlazar Feb 17, 2018
c762a84
Misc cleanup
mjlazar Feb 17, 2018
fd103ca
Handle failure on extended attribute read
mjlazar Feb 17, 2018
fb7694b
Make sure trim methods do not remove objects with overridden age limits
mjlazar Feb 19, 2018
e504b0b
New methods to remove expired objects from cache
mjlazar Feb 19, 2018
24ea58a
Remove expired objects from disk cache on initialization if any objec…
mjlazar Feb 19, 2018
f539d78
Remove expired objects from memory cache on memory warning
mjlazar Feb 19, 2018
a27d755
Fix expiration logic
mjlazar Feb 19, 2018
4b383a5
Add unit test for removeExpiredObjects
mjlazar Feb 19, 2018
662a7cb
Remove assert in -setObject:...
mjlazar Feb 23, 2018
7fa42e3
Call async version of -removeExpiredObjects in -initializeDiskProperties
mjlazar Feb 23, 2018
9c3b33b
Tweaks
mjlazar Feb 23, 2018
7d88344
Merge branch 'master' into object-level-ttl-override
garrettmoon Feb 23, 2018
92e1d59
Fix warnings
mjlazar Feb 24, 2018
71b7fb9
Make ttlCache a readonly property
mjlazar Feb 25, 2018
2d0ae4b
Changes to address feedback from @garrettmoon
mjlazar Mar 17, 2018
6a34a58
Merge branch 'master' into object-level-ttl-override
mjlazar Mar 17, 2018
a47eff1
Merge branch 'master' into object-level-ttl-override
garrettmoon Mar 27, 2018
b5a6640
Changes to address @garrettmoon's feedback
mjlazar Apr 16, 2018
fab908f
Fix indents.
mjlazar Apr 16, 2018
f7216ed
Merge branch 'master' into object-level-ttl-override
mjlazar Apr 19, 2018
60c83aa
Adding additional tests
mjlazar May 8, 2018
2655498
Fix race condition in unit tests
mjlazar May 9, 2018
871f503
Update CHANGELOG
mjlazar May 9, 2018
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
- [fix] Fix up warnings and upgrade to PINOperation 1.1.1: [#213](https://github.com/pinterest/PINCache/pull/213)
- [performance] Reduce locking churn in cleanup methods. [#212](https://github.com/pinterest/PINCache/pull/212)
- [fix] Don't set file protection unless requested. [#220](https://github.com/pinterest/PINCache/pull/220)
- [new] Add ability to set an object level TTL: [#209](https://github.com/pinterest/PINCache/pull/209)

## 3.0.1 -- Beta 6
- [fix] Add some sane limits to the disk cache: [#201]https://github.com/pinterest/PINCache/pull/201
Expand Down
18 changes: 18 additions & 0 deletions PINCache.xcodeproj/project.pbxproj
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,11 @@
6928EED31E4160EE00B5D975 /* PINCaching.h in Headers */ = {isa = PBXBuildFile; fileRef = 6928EED21E4160EE00B5D975 /* PINCaching.h */; settings = {ATTRIBUTES = (Public, ); }; };
6928EED41E4160FE00B5D975 /* PINCaching.h in Headers */ = {isa = PBXBuildFile; fileRef = 6928EED21E4160EE00B5D975 /* PINCaching.h */; settings = {ATTRIBUTES = (Public, ); }; };
6928EED51E41610700B5D975 /* PINCaching.h in Headers */ = {isa = PBXBuildFile; fileRef = 6928EED21E4160EE00B5D975 /* PINCaching.h */; settings = {ATTRIBUTES = (Public, ); }; };
C30EE1F520373D1900D78CB9 /* NSDate+PINCacheTests.m in Sources */ = {isa = PBXBuildFile; fileRef = C30EE1EA203717DE00D78CB9 /* NSDate+PINCacheTests.m */; };
C30EE1F620373D1A00D78CB9 /* NSDate+PINCacheTests.m in Sources */ = {isa = PBXBuildFile; fileRef = C30EE1EA203717DE00D78CB9 /* NSDate+PINCacheTests.m */; };
C30EE1F720373D1B00D78CB9 /* NSDate+PINCacheTests.m in Sources */ = {isa = PBXBuildFile; fileRef = C30EE1EA203717DE00D78CB9 /* NSDate+PINCacheTests.m */; };
C38F01AA20A32E0200F47F0E /* PINDiskCache+PINCacheTests.h in Headers */ = {isa = PBXBuildFile; fileRef = C38F01A820A32E0200F47F0E /* PINDiskCache+PINCacheTests.h */; };
C38F01AB20A32E0200F47F0E /* PINDiskCache+PINCacheTests.m in Sources */ = {isa = PBXBuildFile; fileRef = C38F01A920A32E0200F47F0E /* PINDiskCache+PINCacheTests.m */; };
CC0105DF1E271A5C00890935 /* PINCache.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = CC0105B11E271A1600890935 /* PINCache.framework */; };
CC0105EE1E271A6400890935 /* PINCache.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = CC0105C11E271A4000890935 /* PINCache.framework */; };
CC01060E1E271A9500890935 /* PINCache.m in Sources */ = {isa = PBXBuildFile; fileRef = CC0106061E271A9000890935 /* PINCache.m */; };
Expand Down Expand Up @@ -172,6 +177,10 @@
6818C2901E564C1100875DB7 /* PINOperation.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = PINOperation.xcodeproj; path = Carthage/Checkouts/PINOperation/PINOperation.xcodeproj; sourceTree = "<group>"; };
68A0FBFF1E4D3282000B552D /* PINCacheMacros.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = PINCacheMacros.h; sourceTree = "<group>"; };
6928EED21E4160EE00B5D975 /* PINCaching.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = PINCaching.h; sourceTree = "<group>"; };
C30EE1E9203717DE00D78CB9 /* NSDate+PINCacheTests.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSDate+PINCacheTests.h"; sourceTree = "<group>"; };
C30EE1EA203717DE00D78CB9 /* NSDate+PINCacheTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSDate+PINCacheTests.m"; sourceTree = "<group>"; };
C38F01A820A32E0200F47F0E /* PINDiskCache+PINCacheTests.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "PINDiskCache+PINCacheTests.h"; sourceTree = "<group>"; };
C38F01A920A32E0200F47F0E /* PINDiskCache+PINCacheTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "PINDiskCache+PINCacheTests.m"; sourceTree = "<group>"; };
CC0105B11E271A1600890935 /* PINCache.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = PINCache.framework; sourceTree = BUILT_PRODUCTS_DIR; };
CC0105C11E271A4000890935 /* PINCache.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = PINCache.framework; sourceTree = BUILT_PRODUCTS_DIR; };
CC0105CE1E271A4900890935 /* PINCache.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = PINCache.framework; sourceTree = BUILT_PRODUCTS_DIR; };
Expand Down Expand Up @@ -313,8 +322,12 @@
children = (
CC0106CB1E28248A00890935 /* Assets */,
CC0106C51E281D6900890935 /* Info.plist */,
C30EE1E9203717DE00D78CB9 /* NSDate+PINCacheTests.h */,
C30EE1EA203717DE00D78CB9 /* NSDate+PINCacheTests.m */,
CC0106C91E28228300890935 /* PINCacheTests.h */,
CC01060D1E271A9000890935 /* PINCacheTests.m */,
C38F01A820A32E0200F47F0E /* PINDiskCache+PINCacheTests.h */,
C38F01A920A32E0200F47F0E /* PINDiskCache+PINCacheTests.m */,
);
path = Tests;
sourceTree = "<group>";
Expand All @@ -339,6 +352,7 @@
68A0FC001E4D32C4000B552D /* PINCacheMacros.h in Headers */,
CC0106181E271AAF00890935 /* PINCacheObjectSubscripting.h in Headers */,
CC01061A1E271AAF00890935 /* PINMemoryCache.h in Headers */,
C38F01AA20A32E0200F47F0E /* PINDiskCache+PINCacheTests.h in Headers */,
CC0106191E271AAF00890935 /* PINDiskCache.h in Headers */,
);
runOnlyForDeploymentPostprocessing = 0;
Expand Down Expand Up @@ -643,6 +657,7 @@
files = (
CC01060E1E271A9500890935 /* PINCache.m in Sources */,
CC0106101E271A9500890935 /* PINMemoryCache.m in Sources */,
C38F01AB20A32E0200F47F0E /* PINDiskCache+PINCacheTests.m in Sources */,
CC01060F1E271A9500890935 /* PINDiskCache.m in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
Expand Down Expand Up @@ -672,6 +687,7 @@
buildActionMask = 2147483647;
files = (
CC0106C61E28226900890935 /* PINCacheTests.m in Sources */,
C30EE1F520373D1900D78CB9 /* NSDate+PINCacheTests.m in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
Expand All @@ -680,6 +696,7 @@
buildActionMask = 2147483647;
files = (
CC0106C71E28226A00890935 /* PINCacheTests.m in Sources */,
C30EE1F620373D1A00D78CB9 /* NSDate+PINCacheTests.m in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
Expand All @@ -688,6 +705,7 @@
buildActionMask = 2147483647;
files = (
CC0106C81E28226A00890935 /* PINCacheTests.m in Sources */,
C30EE1F720373D1B00D78CB9 /* NSDate+PINCacheTests.m in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
Expand Down
25 changes: 24 additions & 1 deletion Source/PINCache.h
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,27 @@ PIN_SUBCLASSING_RESTRICTED
serializer:(nullable PINDiskCacheSerializerBlock)serializer
deserializer:(nullable PINDiskCacheDeserializerBlock)deserializer;

/**
Multiple instances with the same name are *not* allowed and can *not* safely
access the same data on disk. Also used to create the <diskCache>.
Initializer allows you to override default NSKeyedArchiver/NSKeyedUnarchiver serialization for <diskCache>.
You must provide both serializer and deserializer, or opt-out to default implementation providing nil values.

@see name
@param name The name of the cache.
@param rootPath The path of the cache on disk.
@param serializer A block used to serialize object before writing to disk. If nil provided, default NSKeyedArchiver serialized will be used.
@param deserializer A block used to deserialize object read from disk. If nil provided, default NSKeyedUnarchiver serialized will be used.
@param keyEncoder A block used to encode key(filename). If nil provided, default url encoder will be used
@param keyDecoder A block used to decode key(filename). If nil provided, default url decoder will be used
@result A new cache with the specified name.
*/
- (instancetype)initWithName:(nonnull NSString *)name
rootPath:(nonnull NSString *)rootPath
serializer:(nullable PINDiskCacheSerializerBlock)serializer
deserializer:(nullable PINDiskCacheDeserializerBlock)deserializer
keyEncoder:(nullable PINDiskCacheKeyEncoderBlock)keyEncoder
keyDecoder:(nullable PINDiskCacheKeyDecoderBlock)keyDecoder;

/**
Multiple instances with the same name are *not* allowed and can *not* safely
Expand All @@ -117,14 +138,16 @@ PIN_SUBCLASSING_RESTRICTED
@param deserializer A block used to deserialize object read from disk. If nil provided, default NSKeyedUnarchiver serialized will be used.
@param keyEncoder A block used to encode key(filename). If nil provided, default url encoder will be used
@param keyDecoder A block used to decode key(filename). If nil provided, default url decoder will be used
@param ttlCache Whether or not the cache should behave as a TTL cache.
@result A new cache with the specified name.
*/
- (instancetype)initWithName:(nonnull NSString *)name
rootPath:(nonnull NSString *)rootPath
serializer:(nullable PINDiskCacheSerializerBlock)serializer
deserializer:(nullable PINDiskCacheDeserializerBlock)deserializer
keyEncoder:(nullable PINDiskCacheKeyEncoderBlock)keyEncoder
keyDecoder:(nullable PINDiskCacheKeyDecoderBlock)keyDecoder NS_DESIGNATED_INITIALIZER;
keyDecoder:(nullable PINDiskCacheKeyDecoderBlock)keyDecoder
ttlCache:(BOOL)ttlCache NS_DESIGNATED_INITIALIZER;

@end

Expand Down
68 changes: 63 additions & 5 deletions Source/PINCache.m
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,17 @@ - (instancetype)initWithName:(NSString *)name
deserializer:(PINDiskCacheDeserializerBlock)deserializer
keyEncoder:(PINDiskCacheKeyEncoderBlock)keyEncoder
keyDecoder:(PINDiskCacheKeyDecoderBlock)keyDecoder
{
return [self initWithName:name rootPath:rootPath serializer:serializer deserializer:deserializer keyEncoder:keyEncoder keyDecoder:keyDecoder ttlCache:NO];
}

- (instancetype)initWithName:(NSString *)name
rootPath:(NSString *)rootPath
serializer:(PINDiskCacheSerializerBlock)serializer
deserializer:(PINDiskCacheDeserializerBlock)deserializer
keyEncoder:(PINDiskCacheKeyEncoderBlock)keyEncoder
keyDecoder:(PINDiskCacheKeyDecoderBlock)keyDecoder
ttlCache:(BOOL)ttlCache
{
if (!name)
return nil;
Expand All @@ -60,7 +71,8 @@ - (instancetype)initWithName:(NSString *)name
deserializer:deserializer
keyEncoder:keyEncoder
keyDecoder:keyDecoder
operationQueue:_operationQueue];
operationQueue:_operationQueue
ttlCache:ttlCache];
_memoryCache = [[PINMemoryCache alloc] initWithOperationQueue:_operationQueue];
}
return self;
Expand Down Expand Up @@ -134,18 +146,28 @@ - (void)setObjectAsync:(id <NSCoding>)object forKey:(NSString *)key completion:(
[self setObjectAsync:object forKey:key withCost:0 completion:block];
}

- (void)setObjectAsync:(id <NSCoding>)object forKey:(NSString *)key withAgeLimit:(NSTimeInterval)ageLimit completion:(PINCacheObjectBlock)block
{
[self setObjectAsync:object forKey:key withCost:0 ageLimit:ageLimit completion:block];
}

- (void)setObjectAsync:(id <NSCoding>)object forKey:(NSString *)key withCost:(NSUInteger)cost completion:(PINCacheObjectBlock)block
{
[self setObjectAsync:object forKey:key withCost:cost ageLimit:0.0 completion:block];
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Once this lands, I wonder if we could reduce the complexity of everything by changing this line to:
[self setObjectAsync:object forKey:key withCost:cost ageLimit:self.isTTLCache ? self.ageLimit : 0.0 completion:block]

Then we could get rid of any other logic that deals with isTTLCache?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think that could certainly be done, but it would have to be in conjunction with some changes to the trim-to-date methods. Since this would essentially set object-level age limits all of the time, we could combine the current trimming logic with the new -removeExpiredObjects logic I'm proposing in this PR.

}

- (void)setObjectAsync:(nonnull id)object forKey:(nonnull NSString *)key withCost:(NSUInteger)cost ageLimit:(NSTimeInterval)ageLimit completion:(nullable PINCacheObjectBlock)block
{
if (!key || !object)
return;

PINOperationGroup *group = [PINOperationGroup asyncOperationGroupWithQueue:_operationQueue];

[group addOperation:^{
[self->_memoryCache setObject:object forKey:key withCost:cost];
[self->_memoryCache setObject:object forKey:key withCost:cost ageLimit:ageLimit];
}];
[group addOperation:^{
[self->_diskCache setObject:object forKey:key];
[self->_diskCache setObject:object forKey:key withAgeLimit:ageLimit];
}];

if (block) {
Expand Down Expand Up @@ -223,6 +245,26 @@ - (void)trimToDateAsync:(NSDate *)date completion:(PINCacheBlock)block
[group start];
}

- (void)removeExpiredObjectsAsync:(PINCacheBlock)block
{
PINOperationGroup *group = [PINOperationGroup asyncOperationGroupWithQueue:_operationQueue];

[group addOperation:^{
[self->_memoryCache removeExpiredObjects];
}];
[group addOperation:^{
[self->_diskCache removeExpiredObjects];
}];

if (block) {
[group setCompletion:^{
block(self);
}];
}

[group start];
}

#pragma mark - Public Synchronous Accessors -

- (NSUInteger)diskByteCount
Expand Down Expand Up @@ -269,13 +311,23 @@ - (void)setObject:(id <NSCoding>)object forKey:(NSString *)key
[self setObject:object forKey:key withCost:0];
}

- (void)setObject:(id <NSCoding>)object forKey:(NSString *)key withAgeLimit:(NSTimeInterval)ageLimit
{
[self setObject:object forKey:key withCost:0 ageLimit:ageLimit];
}

- (void)setObject:(id <NSCoding>)object forKey:(NSString *)key withCost:(NSUInteger)cost
{
[self setObject:object forKey:key withCost:cost ageLimit:0.0];
}

- (void)setObject:(nullable id)object forKey:(nonnull NSString *)key withCost:(NSUInteger)cost ageLimit:(NSTimeInterval)ageLimit
{
if (!key || !object)
return;

[_memoryCache setObject:object forKey:key withCost:cost];
[_diskCache setObject:object forKey:key];
[_memoryCache setObject:object forKey:key withCost:cost ageLimit:ageLimit];
[_diskCache setObject:object forKey:key withAgeLimit:ageLimit];
}

- (nullable id)objectForKeyedSubscript:(NSString *)key
Expand Down Expand Up @@ -310,6 +362,12 @@ - (void)trimToDate:(NSDate *)date
[_diskCache trimToDate:date];
}

- (void)removeExpiredObjects
{
[_memoryCache removeExpiredObjects];
[_diskCache removeExpiredObjects];
}

- (void)removeAllObjects
{
[_memoryCache removeAllObjects];
Expand Down
Loading