Skip to content

Commit

Permalink
Release 5.3.1 (#232)
Browse files Browse the repository at this point in the history
* More PR review changes

* Even more PR review changes

* Removed ConnectionInformationCaching, removed redundant variables

* Simplified synchronizing error behavior for connection status

* DRY up setupListeners

* Fixed string cases for LastConnectionFailureReason Codable

* Forgot to add polling last successful connection

* Store is now in Cache dir, removed listeners and 1 ldclient.shared, changed TimeInterval to Date, changed two inout's to var's

* Added ConnectionModeChangedObserver

* Fixed confusing docstring, removed unnecessary flag synchronizer

* Changed getValue and toString to description

* Changed description to computed property

* Removed connectionModeCheck

* Improved background connection status behavior

* Changed ConnectionInformation description to computed property

* Added new identify method, updated swiftlint rules

* Made Identify actually change the user, fixed unit tests

* Removed unnecessary _user assignment

* Move swiftlint disable to LDClient, move user assignment into identifyInternal

* Added private setOnline and go functions for identify

* Forgot to replace go in guard with goIdentify

* Copied user property unit tests for identify

* Added optimization to not call setOnline when there is no completion and wasOnline is false

* Testing identifyInternal change in IH

* requested PR changes

* Fixed convertCachedData call count, some PR feedback fixes

* Simplified unit tests, added comment about thread safety

* Changed lastSuccessfulConnection to lastKnownFlagValidity, added on stream close listener

* Fixed handler

* Remove unnecessary import

* Changed eventSource access level

* Reset lastKnownFlagValidity to nil when we make a successful stream connection

* Made comment about lastKnownFlagValidity having a value more clear

* Changed guard let to if let in DarklyService EventSource extension

* Updated README, CHANGELOG, and podspec for 4.2.0

* [4.2.1] - 2019-11-15 (#81)

* Added 4.2.1 changes including Dictionary fix and test and eventsource version bump

* Added Carthage build files

* Added attribution to CHANGELOG

* Evaluation Reasons (#82)

* Initial Evaluation Reasons prototype

* Changed evaluationReasons to reasons, fixed mangled question mark in url query parameters

* Added errorKinds

* Added reason to debug events

* Changed reason nil handling in events

* Removed unnecessary private functions, removed event nil coalesce

* Event change for reason nil

* Removed if in Event for reason nil, added guard let NSNull for dictionary in Events

* Added reason null to normal variation calls

* Testing alternate reason null set

* Change Event comparison code to add reason

* Removed reason from event dictionary matches

* Removed Event Equatable reason and changed Event Dictionary matches

* Explicitly set featureflag reason property to nil when not send feature event for variationDetail

* Fixed DeprecatedCacheModel unit tests, added v6 model tests, added variation detail unit tests, changed event reason nil check

* Added variationDetail to objective c, fixed cache converter unit tests, deprecated variationAndSource

* Added new files to correct target membership

* Simplify featureflag reason nil check

* Removed unnecessary comments, deprecated objc variationAndSource

* Improved doc strings, cut down on variation doc string repetition, added reason tests for Event and EventReporter, added EvaluationDetail generic

* Added ObjCEvaluationDetail to correct target membership

* Differentiated objc evaluation detail return type, change reason constant in unit tests

* Removed Optional value on fallback variation

* Added Event test for reason false but flag reason present, cleaned up objc doc strings, changed optional types in objc evaluation detail

* Experimentation 1.5 (#83)

* Added trackReasons and metricValue as well as doc and unit tests

* Added unit tests for reason false track reason true, added objc track without metricValue, changed and trackReason to or trackReason

* Added explicit event store reason test

* Fixed objc track comment

* Updated reason to includeReason for clarity

* Explicitly check both feature and debug events

* Prepare version 4.3.0

* Update ios-eventsource to 4.1.0 (#84)

* Updated ios-eventsource to 4.1.0, updated version strings

* Added sentence about iOS SDK compat to CHANGELOG

* pod repo update to get new eventsource version

* Fixed change listener not firing when only value changes (#85)

* Fixed change listener not firing when only value changes

* Fixed a unit test variable name typo

* Cast flag value to dictionary instead of string, improved unit test

* Remove debug printout, remove unnecessary parameter

* Updated CHANGELOG, README, and podspec for 4.3.2

* Adds startAwaitingFlags function (#86)

* Added newStart function, updated README, podspec, and CHANGELOG

* Changed newStart to startAwaitingFlags

* Added doc strings for startAwaitingFlags and deprecated start

* Added cocoapods lib lint flag to ignore depracted API usage in ObjCLDClient, added startAwaitingFlags to ObjC

* Fixed alow to allow, changed startAwaitFlags to startCompleteWhenFlagsReceived, added unit tests for startCompleteWhenFlagsReceived

* Updated deprecation message to startCompleteWhenFlagsReceived

* [ch61092] Add X-LaunchDarkly-Payload-ID to event request headers (#87)

* Added X-LaunchDarkly-Payload-ID header to event request headers, add string length check on LD payload ID header in header unit tests

* Added CHANGELOG Added entry for event request header

* Added UUID generation to DarklyService so it's regenerated on each request

* Change CHANGELOG entry for new header to match other SDKs

* Added single retry attempt to event post (#88)

* Added single retry attempt to event post

* Added payloadId to the correct point to maintain value between retries, retry on error

* reportSyncComplete on 2nd error, always log on event post error, add 1 second delay with log on event post retry

* Added 1 second delay to retry event post

* async'd event post to prevent blocking main thread on retry sleep

* Increased event test waitUntil timeout to 10 seconds

* Bumped patch version, added CHANGELOG for 4.4.1

* Changed 4.4.1 entry

* Further 4.4.1 entry revision

* Changed retry event post for loop to function call (#89)

* [ch65670] Fix Xcode 11.4 build and unit tests (#90)

* Fixed build errors, fixed some unit tests

* Added shell script to remove duplicate sourcery method mock

* sed now matches specific pattern instead of line numbers

* Add project config file containing run script step

* Bump CircleCI Xcode to 11.4

* Bump CircleCI Xcode to 11.4.0

* Changed iPhone simulator in CircleCI from XS to Xs

* Changed CircleCI xcodebuild from OS:latest to OS:12.4

* Testing NSURLRetryPolicy on CI

* Removed DeprecatedCache from automock

* Fix FlagStore to synchronize reads with writes.

* multi env: init, config, client methods, and ObjCLDClient

* Remove accidentally checked in IDE file

* Add xcode compatibility information to readme (#93)

* Add xcode compatibility information to readme (#93)

* Fixed comments, added flush to close, started migrating unit tests

* Added multi environment methods to ObjCLDClient, added better comments for said methods

* Changed init completion behavior for correct ordering, removed deprecated variationAndSource methods, more unit test changes

* Added primary instance to instances, fixed completion behavior

* Made UserEnvironmentFlagCache shared between instances, made FlagSynchronizer DispatchQueue not static

* Fixed user clobbering by deep copying FlagStore

* Fix for concurrency bug exposed by restwrapper. (#94)

* General cleanup. (#92)

* [ch75316] Added maxCachedUsers to LDConfig and UserEnvironmentFlagCache (#96)

* Added maxCachedUsers to LDConfig and UserEnvironmentFlagCache

* Changed specific -1 to 0

* Added unlimited users tests

* Added CHANGELOG for 4.6

* Added maxCachedUsers to objc LDConfig (#97)

* Changed version to 4.6.0

* [ch66864] Added startCompleteWhenFlagsReceivedTimeout (#98)

* Added startCompleteWhenFlagsReceivedTimeout which allows a timeout in seconds to be given for the completion to be fired indicating that flags have been received

* Fixed doc typo, removed bad ms to second conversion

* Fix previous doc typo that got copied into objc

* Added serial queue to timeOutCheck to fix possible race condition

* Simplified unit tests by swapping if for assertion

* Simplified start timeout method name, fixed doc about timedOut return

* Moved DispatchQueue off of main

* Moved sync dispatchqueue from variable to func

* Changed labeled queue to global queue

* Added test for start timeout timing out, added doc note about start timeout and starting offline

* [ch70792] Swift EventSource (#99)

* Integrating Swift eventsource.

* No longer set `-sdk` on `xcodebuild` commands for circle. There seems
to be some sort of interaction between `-sdk` or `SDKROOT` and the
implicit dependency resolution system. Since Xcode 10.2, Xcode can
find implicit dependencies in `OTHER_LINKER_FLAGS` such as when
Cocoapods links a framework with `-F`. When `SDKROOT` is not set, this
doesn't conflict with anything, but I believe the behavior of finding
the framework as an implicit dependency is not required (I think it
just resolves to the same target Cocoapods is building). If `SDKROOT`
is set, xcodebuild seems to decide that it must first resolve and
build the framework. It gives the warning "Multiple targets match
implicit dependency for linker flags '-framework LDSwiftEventSource'",
as LDSwiftEventSource generates a product by the same name
(`LDSwiftEventSource.framework`) regardless of the platform/sdk being
built. Now apparently it completely forgets about the set `SDKROOT`
(which normally seems to be fed through and used to resolve the target
being built) and decides that it should build the iOS target (no idea
why, it's not even the default `SDKROOT` for
`LDSwiftEventSource`). When `SDKROOT` is set to `macosx`, `xcodebuild`
proceeds to simultaneously build both the iOS and macOS targets (I
believe the macOS target is from Cocoapods operating correctly), then
fails as it's still actually building the iOS target with the macOS
SDK configuration. The actual build error is that the
'LDSwiftEventSource-iOS-umbrella.h' header can't find 'UIKit/UIKit.h'
as UIKit is not available on macOS.

* Update LDSwiftEventSource to 0.3.0, handle some minor todos.

* This updates the circle config to actually do a `pod install`. Removes
Gemfile that was for CocoaPods.

* Specify sources for pod spec lint.

* I think spec lint tries to test the spec file against code on the
master branch, so only run a spec lint if commit is on master.

* Allow warnings for now.

* Bump version to 4.7.0 and add CHANGELOG

* Changed startWaitSeconds type signature from Int to TimeInterval

* Changed startWaitSeconds doc string type to TimeInterval

* Clarified 4.7.0 CHANGELOG

* Fixed no log output when isDebugMode = true

* Fixed old EventSource still sticking around, removed default to fix compiler warning

* Added startWaitSeconds to start, added more debug logging, added flagsynchronizer config.streamingMode

* Fixed LDClient offline by putting internalIdentify in an async DispatchQueue, removed print statements

* Unit tests build but LDClientSpec crashes

* [ch72204] drop events after failed retry (#100)

* Drop events after failed retry, as well as some general cleanup and
consistency changes.

* Don't disabled event reporting in background.

* Added ClientServiceFactory back to init

* Throttler mock isn't firing, migrating unit tests to TestContext init pattern

* [ch61747] Diagnostic events (#101)

* Start specs build and run but fail

* Removed public set LDClient.config, added more failing specs, removed prints

* Removed convenience init flagSynchronizer set to avoid malloc error, all but 2 LDClient specs build and run

* All LDClient specs build and run

* Fixed LDClient spec flagstore mock being replaced

* Remove EventTrackingContext

* Added config didSet back to init, removed outdated tests

* Removed outdated tests for client not started, fixed some incorrectly ported tests

* Fix more LDClient unit tests failures

* Remove EventTrackingContext (#102)

* Remove DeprecatedCacheModelV6, which was never needed.

* A bit of cleanup.

* Update some old comments.

* Update diagnostic events to include count of secondary mobile keys.

* Added waitUntil to internalIdentify calls to fix timing issue in some tests

* Added setOnline for background tests, removed unnecessary runMode parameter on test start

* set hasStarted false for starting tests

* Fixed static timeOutCheck not being reset, trigger service didSet for pollingInterval tests

* Added secondaryMobileKeys to ObjCLDConfig, change Array<String>? to [String]?, changed getEnvNames return, removed unnecessary returns in ClientServiceFactory, changed to internalSetOnline in close

* Set instances to nil after loop in close

* Remove UUID queue labels

* Forgot empty FlagStore init

* Added waitUntil to identify to fix Circle timing

* Only internal ObjC methods can set the ObjCLDClient.ldClient

* Moved all internalIdentify calls to waitUntil to fix Circle timing

* Make config public private(set)

* Added debug method to set config allUserAttributesPrivate

* Make testing allUserAttributesPrivate method public

* Move setAllUserAttributesPrivate to public

* Attempted timing fix for Circle

* Lowered timeout with troublesome test

* Troublesome test false to true

* Try synchronized instead of DispatchQueue.sync

* Change sync queue and if ordering in start timeout

* Forgot to add to testing version

* Removed spurious changes from FlagSynchronizer and FlagStore

* Try synced with if change start timeout

* Changed setAllUserAttributesPrivate to apply to all instances, changed setOnline to internalSetOnline in configDidSet

* Changed LDClient internal setOnline calls to internalSetOnline

* Removed synced, added serial queues to identify and setonline

* Removed true check in start timeout

* Test removing serial queue from setOnline and identify

* Change serial queues in setOnline and identify to sync

* Move if outside sync serial queue in start timeout

* Move if back inside serial queue but make queue async start timeout

* remove static from internal method queues

* Add secondaryMobileKeys to convertCacheData

* Switched global to main queue in start timeout

* Try name queue instead of main in start timeout

* Remove internalIdentifyQueue

* Changed named queue back to global, added back InternalIdentifyQueue, changed CircleCi simulator from iOS 12.4 to 13.4

* Change iOS 13.4 to 13.4.1 in Circle

* Change iOS simulator from Xs to iPhone 11 Pro Max

* Updating to iOS 13.4.1 on Circle fix disparity between DarklyServiceSpec on local vs Circle

* Removed static from timeOutCheck and queue and made them function scope variables

* Made main queues named queues in LDClientSpec

* Explicitly turn off parallel testing on Circle

* Added waitUntil with asyncAfter to setRunMode and background notification

* Changed all asyncAfter deadlines to 0.2

* Passes thread sanitizer locally

* More thread sanitizer finds

* Fixed recursive sync queues in EventReporter and FlagSynchronizer

* Improve Throttler code.

* Hopefully fix sync queue crash in FlagChangeNotifier.

* I don't think we actually want to loop through converting all
environment cache data here.

* changed internalIdentifyQueue to sync, added sync queue to internalSetOnline

* PR feedback, removed convenience init

* Fixed onSuccess tests

* Removes isStarting variable, to make initialization more clear.

* Removes all remaining LDFlagSource code.

* Fix logging of type conversion errors.

* Update more code doc for multi-environment branch changes.

* Remove unnecessary objc class and improve interface for type
mismatches on changed flag handlers.

* Prefer `is` to `as != nil`.

* Use DispatchGroup for static loops.

* Make time constraint a bit more lenient for more reliable CircleCI
tests.

* Add a threshold for testing delay on throttler executions to prevent
spurious test failures.

* Replaced timeOutCheck with completed, simplified start test close, removed unnecessary string cast in LDConfig

* Removed unnecessary nil check in start for tests

* Removed unnecessary string cast in LDConfig, made waitUntil longer than start timeout

* Fix after merge.

* Change setUserSpec to identifySpec, fix doc strings

* Remove internalInstances from LDClient, remove isOnlineQueue from DiagnosticReporter

* Should improve LDClientSpec test runtime substantially.

* 3 second for timeout seems a bit long even for circleci.

* Update OHHTTPStubs for SwiftPM support.

* Update DarklyServiceMock for changes in OHHTTPStubs 9.0.0

* Update DarklyServiceSpec for OHHTTPStubs 9.0.0

* Add explicit import for Foundation in tests. Make linter slightly more
lienent and specify linted paths better. Fix mocks.stencil to allow
specifying a defaultMockValue for an Optional type variable. Change
default value for EnvironmentReporting.backgroundNotification and
EnvironmentReporting.foregroundNotification to delegate to
EnvironmentReporter to handle running tests on platforms without
UIKit. Avoid using XCTFail.

* Add initial SwiftPM Package.swift definition. Add SwiftPM build folder
to .gitignore. Add missing Foundation import in
LDClientSpec.swift. Comment out test that was previously disabled in
the xcode test configuration, which SwiftPM doesn't use.

* Use github reference for LDSwiftEventSource rather than local
reference.

* Try specifying ssh urls for dependencies in Package.swift as xcode is
choking when adding the package.

* Remove setAllUserAttributesPrivate public method

* Changed ObjcLDClient to act as a wrapper instead of keeping an internal LDClient, updated doc strings

* Fixed weird setOnline tests, fixed doc strings, added throws setter to secondaryMobileKeys, fixed ObjcLDClient

* Fixed LDClient close tests, added getter setter doc strings for secondaryMobileKeys, removed commented out code in LDConfig, removed unnecessary setOnline method in ObjcLDClient

* Fix doc strings in LDClient

* Removed logs from throws in LDConfig

* Added .swiftpm/ to .gitignore, removed all occurrences of fallback and replaced them with defaultValue (#111)

* Added .swiftpm/ to .gitignore, removed all occurrences of fallback and replaced them with defaultValue

* Change .swiftpm/ to /.swiftpm

* Removed defaultValue value

* Fix 2 typos

* Updated SDK version and supported iOS and Xcode versions in README

* Use Mint for tooling.

* Update CONTRIBUTING.md

Co-authored-by: Ben Woskow <[email protected]>

* Add @objc annotation to start objc bridge methods.

* (5.0) Use SwiftPM for SDK dependencies. (#114)

* (5.0) Fix of crash when logging is enabled. (#115)

Also removed unused test code. Improve documentation on LDClient.flush()

* (5.0) Update installation instructions (#116)

* (5.0) Changelog and some final removals and renames of public APIs (#117)

Fix a bug with preventing the SDK from retrying eventsource connections when the service returns a 5XX error code. Renamed a couple of public APIs. Removed some publicly exposed implementation details. Updated CHANGELOG.md for 5.0.

* [ch69438] Add ability to set additional headers on requests (#119)

* A jazzy doc build and update lint configuration (#120)

* [ch84954] Fix nested bundles in carthage build and missing dynamic library in SwiftPM build. (#121)

* Focus week cleanup (#122)

* [ch87108] Fix LDUser serialization to report redacted custom attributes in all cases (#123)

* [ch87107] Add LDUser secondary attribute (#124)

* [ch88282] Fix DiagnosticEvent build warnings on Xcode 12 (#125)

* Update dependencies after release of Nimble 9.0.0 (#126)

* Update doc comments for LDClient.identify (#127)

* [ch91505] Dynamic header configuration (#128)

Allow dynamic configuration of http headers through LDConfig.headerDelegate

* [ch94946] Multi Environment Bug Fix: Decouple FlagStore from LDUser (#130)

* Removed FlagStore from LDUser

* All unit tests build and LDClientSpec doesn't crash

* Fixed 1 unit test, removed unnecessary testing parameter in internalIdentify

* Fixed 29 unit tests

* Fixed 26 unit tests

* Fix all but 4 unit tests

* Removed comment

* Addressed PR feedback

* Removed unnecessary for each in LDUserSpec

* Removed unnecessary contexts in LDUserSpec, removed user value in FlagChangeNotifierSpec

* Fixed LDUserSpec context, added stub flags to DarklyServiceSpec

* Make LDClientSpec init less hacky

* Clear FlagStore if new user does not have cached flags, make flagStore a let

* Added FlagStore back to LDUser for init from flag dictionary, check for user.flagStore on LDClient init and identify

* Added deprecation comment to dictionary init for LDUser, prioritize LDUser FlagStore over cache in identify

* Changed identify ordering back to cache over flagstore, made FlagStore a let

* Bump version and CHANGELOG for 5.3.1

Co-authored-by: Gavin Whelan <[email protected]>
Co-authored-by: Ben Woskow <[email protected]>
Co-authored-by: Elliot <[email protected]>
  • Loading branch information
4 people authored Dec 16, 2020
1 parent 5481134 commit 89dcc5d
Show file tree
Hide file tree
Showing 22 changed files with 272 additions and 290 deletions.
4 changes: 4 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,10 @@

All notable changes to the LaunchDarkly iOS SDK will be documented in this file. This project adheres to [Semantic Versioning](http://semver.org).

## [5.3.1] - 2020-12-15
### Fixed
- Decoupled `FlagStore` from `LDUser` to fix a bug where multiple environments could overwrite each other's flag values.

## [5.3.0] - 2020-11-06
### Added
- Adds to `LDConfig` the ability to dynamically configure the HTTP headers on requests through the `headerDelegate` property, which has the type `RequestHeaderTransform`.
Expand Down
2 changes: 1 addition & 1 deletion LaunchDarkly.podspec
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
Pod::Spec.new do |ld|

ld.name = "LaunchDarkly"
ld.version = "5.3.0"
ld.version = "5.3.1"
ld.summary = "iOS SDK for LaunchDarkly"

ld.description = <<-DESC
Expand Down
20 changes: 10 additions & 10 deletions LaunchDarkly.xcodeproj/project.pbxproj
Original file line number Diff line number Diff line change
Expand Up @@ -1591,7 +1591,7 @@
GCC_C_LANGUAGE_STANDARD = gnu11;
INFOPLIST_FILE = "$(PROJECT_DIR)/LaunchDarkly/LaunchDarkly/Support/Info.plist";
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
MARKETING_VERSION = 5.3.0;
MARKETING_VERSION = 5.3.1;
MODULEMAP_FILE = "";
PRODUCT_BUNDLE_IDENTIFIER = "com.launchdarkly.Darkly-tvOS";
PRODUCT_NAME = LaunchDarkly_tvOS;
Expand All @@ -1614,7 +1614,7 @@
GCC_C_LANGUAGE_STANDARD = gnu11;
INFOPLIST_FILE = "$(PROJECT_DIR)/LaunchDarkly/LaunchDarkly/Support/Info.plist";
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
MARKETING_VERSION = 5.3.0;
MARKETING_VERSION = 5.3.1;
MODULEMAP_FILE = "";
PRODUCT_BUNDLE_IDENTIFIER = "com.launchdarkly.Darkly-tvOS";
PRODUCT_NAME = LaunchDarkly_tvOS;
Expand All @@ -1637,7 +1637,7 @@
GCC_C_LANGUAGE_STANDARD = gnu11;
INFOPLIST_FILE = "$(PROJECT_DIR)/LaunchDarkly/LaunchDarkly/Support/Info.plist";
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
MARKETING_VERSION = 5.3.0;
MARKETING_VERSION = 5.3.1;
PRODUCT_BUNDLE_IDENTIFIER = "com.launchdarkly.Darkly-macOS";
PRODUCT_NAME = LaunchDarkly_macOS;
SDKROOT = macosx;
Expand All @@ -1658,7 +1658,7 @@
GCC_C_LANGUAGE_STANDARD = gnu11;
INFOPLIST_FILE = "$(PROJECT_DIR)/LaunchDarkly/LaunchDarkly/Support/Info.plist";
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
MARKETING_VERSION = 5.3.0;
MARKETING_VERSION = 5.3.1;
PRODUCT_BUNDLE_IDENTIFIER = "com.launchdarkly.Darkly-macOS";
PRODUCT_NAME = LaunchDarkly_macOS;
SDKROOT = macosx;
Expand Down Expand Up @@ -1702,7 +1702,7 @@
CURRENT_PROJECT_VERSION = 1;
DEBUG_INFORMATION_FORMAT = dwarf;
DYLIB_COMPATIBILITY_VERSION = 5.3.0;
DYLIB_CURRENT_VERSION = 5.3.0;
DYLIB_CURRENT_VERSION = 5.3.1;
ENABLE_STRICT_OBJC_MSGSEND = YES;
ENABLE_TESTABILITY = YES;
FRAMEWORK_VERSION = B;
Expand Down Expand Up @@ -1773,7 +1773,7 @@
CURRENT_PROJECT_VERSION = 1;
DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
DYLIB_COMPATIBILITY_VERSION = 5.3.0;
DYLIB_CURRENT_VERSION = 5.3.0;
DYLIB_CURRENT_VERSION = 5.3.1;
ENABLE_NS_ASSERTIONS = NO;
ENABLE_STRICT_OBJC_MSGSEND = YES;
FRAMEWORK_VERSION = B;
Expand Down Expand Up @@ -1812,7 +1812,7 @@
INFOPLIST_FILE = "$(PROJECT_DIR)/LaunchDarkly/LaunchDarkly/Support/Info.plist";
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
LD_DYLIB_INSTALL_NAME = "$(DYLIB_INSTALL_NAME_BASE:standardizepath)/$(EXECUTABLE_PATH)";
MARKETING_VERSION = 5.3.0;
MARKETING_VERSION = 5.3.1;
MODULEMAP_FILE = "$(PROJECT_DIR)/Framework/module.modulemap";
PRODUCT_BUNDLE_IDENTIFIER = com.launchdarkly.Darkly;
PRODUCT_NAME = LaunchDarkly;
Expand All @@ -1832,7 +1832,7 @@
INFOPLIST_FILE = "$(PROJECT_DIR)/LaunchDarkly/LaunchDarkly/Support/Info.plist";
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
LD_DYLIB_INSTALL_NAME = "$(DYLIB_INSTALL_NAME_BASE:standardizepath)/$(EXECUTABLE_PATH)";
MARKETING_VERSION = 5.3.0;
MARKETING_VERSION = 5.3.1;
MODULEMAP_FILE = "$(PROJECT_DIR)/Framework/module.modulemap";
PRODUCT_BUNDLE_IDENTIFIER = com.launchdarkly.Darkly;
PRODUCT_NAME = LaunchDarkly;
Expand Down Expand Up @@ -1874,7 +1874,7 @@
GCC_C_LANGUAGE_STANDARD = gnu11;
INFOPLIST_FILE = "$(PROJECT_DIR)/LaunchDarkly/LaunchDarkly/Support/Info.plist";
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
MARKETING_VERSION = 5.3.0;
MARKETING_VERSION = 5.3.1;
PRODUCT_BUNDLE_IDENTIFIER = "com.launchdarkly.Darkly-watchOS";
PRODUCT_NAME = LaunchDarkly_watchOS;
SDKROOT = watchos;
Expand All @@ -1896,7 +1896,7 @@
GCC_C_LANGUAGE_STANDARD = gnu11;
INFOPLIST_FILE = "$(PROJECT_DIR)/LaunchDarkly/LaunchDarkly/Support/Info.plist";
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
MARKETING_VERSION = 5.3.0;
MARKETING_VERSION = 5.3.1;
PRODUCT_BUNDLE_IDENTIFIER = "com.launchdarkly.Darkly-watchOS";
PRODUCT_NAME = LaunchDarkly_watchOS;
SDKROOT = watchos;
Expand Down
6 changes: 3 additions & 3 deletions LaunchDarkly/GeneratedCode/mocks.generated.swift
Original file line number Diff line number Diff line change
Expand Up @@ -453,10 +453,10 @@ final class FlagChangeNotifyingMock: FlagChangeNotifying {
// MARK: notifyObservers
var notifyObserversCallCount = 0
var notifyObserversCallback: (() -> Void)?
var notifyObserversReceivedArguments: (user: LDUser, oldFlags: [LDFlagKey: FeatureFlag])?
func notifyObservers(user: LDUser, oldFlags: [LDFlagKey: FeatureFlag]) {
var notifyObserversReceivedArguments: (flagStore: FlagMaintaining, oldFlags: [LDFlagKey: FeatureFlag])?
func notifyObservers(flagStore: FlagMaintaining, oldFlags: [LDFlagKey : FeatureFlag]) {
notifyObserversCallCount += 1
notifyObserversReceivedArguments = (user: user, oldFlags: oldFlags)
notifyObserversReceivedArguments = (flagStore: flagStore, oldFlags: oldFlags)
notifyObserversCallback?()
}
}
Expand Down
43 changes: 25 additions & 18 deletions LaunchDarkly/LaunchDarkly/LDClient.swift
Original file line number Diff line number Diff line change
Expand Up @@ -267,20 +267,22 @@ public class LDClient {
}
}

func internalIdentify(newUser: LDUser, testing: Bool = false, completion: (() -> Void)? = nil) {
func internalIdentify(newUser: LDUser, completion: (() -> Void)? = nil) {
internalIdentifyQueue.sync {
var internalUser = newUser
if !testing {
internalUser.flagStore = FlagStore(featureFlagDictionary: newUser.flagStore.featureFlags)
}
self.user = internalUser
self.user = newUser
Log.debug(self.typeName(and: #function) + "new user set with key: " + self.user.key )
let wasOnline = self.isOnline
self.internalSetOnline(false)

cacheConverter.convertCacheData(for: user, and: config)
if let cachedFlags = self.flagCache.retrieveFeatureFlags(forUserWithKey: self.user.key, andMobileKey: self.config.mobileKey), !cachedFlags.isEmpty {
self.user.flagStore.replaceStore(newFlags: cachedFlags, completion: nil)
flagStore.replaceStore(newFlags: cachedFlags, completion: nil)
} else {
if let userFlagStore = user.flagStore {
flagStore.replaceStore(newFlags: userFlagStore.featureFlags, completion: nil)
} else {
flagStore.replaceStore(newFlags: [:], completion: nil)
}
}
self.service = self.serviceFactory.makeDarklyServiceProvider(config: self.config, user: self.user)
self.service.clearFlagResponseCache()
Expand Down Expand Up @@ -368,7 +370,7 @@ public class LDClient {
- returns: LDEvaluationDetail which wraps the requested feature flag value, or the default value, which variation was served, and the evaluation reason.
*/
public func variationDetail<T: LDFlagValueConvertible>(forKey flagKey: LDFlagKey, defaultValue: T) -> LDEvaluationDetail<T> {
let featureFlag = user.flagStore.featureFlag(for: flagKey)
let featureFlag = flagStore.featureFlag(for: flagKey)
let reason = checkErrorKinds(featureFlag: featureFlag) ?? featureFlag?.reason
let value = variationInternal(forKey: flagKey, defaultValue: defaultValue, includeReason: true)
return LDEvaluationDetail(value: value ?? defaultValue, variationIndex: featureFlag?.variation, reason: reason)
Expand Down Expand Up @@ -439,7 +441,7 @@ public class LDClient {
- returns: LDEvaluationDetail which wraps the requested feature flag value, or the default value, which variation was served, and the evaluation reason.
*/
public func variationDetail<T: LDFlagValueConvertible>(forKey flagKey: LDFlagKey, defaultValue: T? = nil) -> LDEvaluationDetail<T?> {
let featureFlag = user.flagStore.featureFlag(for: flagKey)
let featureFlag = flagStore.featureFlag(for: flagKey)
let reason = checkErrorKinds(featureFlag: featureFlag) ?? featureFlag?.reason
let value = variationInternal(forKey: flagKey, defaultValue: defaultValue, includeReason: true)
return LDEvaluationDetail(value: value, variationIndex: featureFlag?.variation, reason: reason)
Expand All @@ -451,7 +453,7 @@ public class LDClient {
Log.debug(typeName(and: #function) + "returning defaultValue: \(defaultValue.stringValue)." + " LDClient not started.")
return defaultValue
}
let featureFlag = user.flagStore.featureFlag(for: flagKey)
let featureFlag = flagStore.featureFlag(for: flagKey)
let value = (featureFlag?.value as? T) ?? defaultValue
let failedConversionMessage = self.failedConversionMessage(featureFlag: featureFlag, defaultValue: defaultValue)
Log.debug(typeName(and: #function) + "flagKey: \(flagKey), value: \(value.stringValue), defaultValue: \(defaultValue.stringValue), featureFlag: \(featureFlag.stringValue), reason: \(featureFlag?.reason?.description ?? "No evaluation reason")."
Expand Down Expand Up @@ -491,7 +493,7 @@ public class LDClient {
public var allFlags: [LDFlagKey: Any]? {
guard hasStarted
else { return nil }
return user.flagStore.featureFlags.allFlagValues
return flagStore.featureFlags.allFlagValues
}

// MARK: Observing Updates
Expand Down Expand Up @@ -668,19 +670,19 @@ public class LDClient {
Log.debug(typeName(and: #function) + "result: \(result)")
switch result {
case let .success(flagDictionary, streamingEvent):
let oldFlags = user.flagStore.featureFlags
let oldFlags = flagStore.featureFlags
connectionInformation = ConnectionInformation.checkEstablishingStreaming(connectionInformation: connectionInformation)
switch streamingEvent {
case nil, .ping?, .put?:
user.flagStore.replaceStore(newFlags: flagDictionary) {
flagStore.replaceStore(newFlags: flagDictionary) {
self.updateCacheAndReportChanges(user: self.user, oldFlags: oldFlags)
}
case .patch?:
user.flagStore.updateStore(updateDictionary: flagDictionary) {
flagStore.updateStore(updateDictionary: flagDictionary) {
self.updateCacheAndReportChanges(user: self.user, oldFlags: oldFlags)
}
case .delete?:
user.flagStore.deleteFlag(deleteDictionary: flagDictionary) {
flagStore.deleteFlag(deleteDictionary: flagDictionary) {
self.updateCacheAndReportChanges(user: self.user, oldFlags: oldFlags)
}
}
Expand All @@ -702,8 +704,8 @@ public class LDClient {

private func updateCacheAndReportChanges(user: LDUser,
oldFlags: [LDFlagKey: FeatureFlag]) {
flagCache.storeFeatureFlags(user.flagStore.featureFlags, forUser: user, andMobileKey: config.mobileKey, lastUpdated: Date(), storeMode: .async)
flagChangeNotifier.notifyObservers(user: user, oldFlags: oldFlags)
flagCache.storeFeatureFlags(flagStore.featureFlags, forUser: user, andMobileKey: config.mobileKey, lastUpdated: Date(), storeMode: .async)
flagChangeNotifier.notifyObservers(flagStore: flagStore, oldFlags: oldFlags)
}

// MARK: Events
Expand Down Expand Up @@ -880,6 +882,7 @@ public class LDClient {
private(set) var environmentReporter: EnvironmentReporting
private(set) var throttler: Throttling
private(set) var diagnosticReporter: DiagnosticReporting
let flagStore: FlagMaintaining

private(set) var hasStarted: Bool {
get { hasStartedQueue.sync { _hasStarted } }
Expand All @@ -894,6 +897,10 @@ public class LDClient {
}
environmentReporter = self.serviceFactory.makeEnvironmentReporter()
flagCache = newCache
flagStore = self.serviceFactory.makeFlagStore()
if let userFlagStore = startUser?.flagStore {
flagStore.replaceStore(newFlags: userFlagStore.featureFlags, completion: nil)
}
LDUserWrapper.configureKeyedArchiversToHandleVersion2_3_0AndOlderUserCacheFormat()
cacheConverter = self.serviceFactory.makeCacheConverter(maxCachedUsers: configuration.maxCachedUsers)
flagChangeNotifier = flagNotifier
Expand Down Expand Up @@ -931,7 +938,7 @@ public class LDClient {
Log.level = environmentReporter.isDebugBuild && config.isDebugMode ? .debug : .noLogging
cacheConverter.convertCacheData(for: user, and: config)
if let cachedFlags = flagCache.retrieveFeatureFlags(forUserWithKey: user.key, andMobileKey: config.mobileKey), !cachedFlags.isEmpty {
user.flagStore.replaceStore(newFlags: cachedFlags, completion: nil)
flagStore.replaceStore(newFlags: cachedFlags, completion: nil)
}

eventReporter.record(Event.identifyEvent(user: user))
Expand Down
20 changes: 4 additions & 16 deletions LaunchDarkly/LaunchDarkly/Models/User/LDUser.swift
Original file line number Diff line number Diff line change
Expand Up @@ -4,16 +4,12 @@
//
// Copyright © 2017 Catamorphic Co. All rights reserved.
//

import Foundation

typealias UserKey = String //use for identifying semantics for strings, particularly in dictionaries

/**
LDUser allows clients to collect information about users in order to refine the feature flag values sent to the SDK. For example, the client app may launch with the SDK defined anonymous user. As the user works with the client app, information may be collected as needed and sent to LaunchDarkly. The client app controls the information collected, which LaunchDarkly does not use except as the client directs to refine feature flags. Client apps should follow [Apple's Privacy Policy](apple.com/legal/privacy) when collecting user information.

The SDK caches last known feature flags for use on app startup to provide continuity with the last app run. Provided the LDClient is online and can establish a connection with LaunchDarkly servers, cached information will only be used a very short time. Once the latest feature flags arrive at the SDK, the SDK no longer uses cached feature flags. The SDK retains feature flags on the last 5 client defined users. The SDK will retain feature flags until they are overwritten by a different user's feature flags, or until the user removes the app from the device.

The SDK does not cache user information collected, except for the user key. The user key is used to identify the cached feature flags for that user. Client app developers should use caution not to use sensitive user information as the user-key.
*/
public struct LDUser {
Expand All @@ -26,9 +22,7 @@ public struct LDUser {

/**
LDUser attributes that can be marked private.

The SDK will not include private attribute values in analytics events, but private attribute names will be sent.

See Also: `LDConfig.allUserAttributesPrivate`, `LDConfig.privateUserAttributes`, and `privateAttributes`.
*/
public static var privatizableAttributes: [String] { optionalAttributes + [CodingKeys.custom.rawValue] }
Expand Down Expand Up @@ -70,24 +64,19 @@ public struct LDUser {
public var operatingSystem: String?
/**
Client app defined privateAttributes for the user.

The SDK will not include private attribute values in analytics events, but private attribute names will be sent.

This attribute is ignored if `LDConfig.allUserAttributesPrivate` is true. Combined with `LDConfig.privateUserAttributes`. The SDK considers attributes appearing in either list as private. Client apps may define attributes found in `privatizableAttributes` and top level `custom` dictionary keys here. (Default: nil)

See Also: `LDConfig.allUserAttributesPrivate` and `LDConfig.privateUserAttributes`.

*/
public var privateAttributes: [String]?

///An NSObject wrapper for the Swift LDUser struct. Intended for use in mixed apps when Swift code needs to pass a user into an Objective-C method.
public var objcLdUser: ObjcLDUser { ObjcLDUser(self) }

internal var flagStore: FlagMaintaining = FlagStore()
internal var flagStore: FlagMaintaining?

/**
Initializer to create a LDUser. Client configurable attributes each have an optional parameter to facilitate setting user information into the LDUser. The SDK will automatically set `key`, `device`, `operatingSystem`, and `isAnonymous` attributes if the client does not provide them. The SDK embeds `device` and `operatingSystem` into the `custom` dictionary for transmission to LaunchDarkly.

- parameter key: String that uniquely identifies the user. If the client app does not define a key, the SDK will assign an identifier associated with the anonymous user.
- parameter name: Client app defined name for the user. (Default: nil)
- parameter firstName: Client app defined first name for the user. (Default: nil)
Expand Down Expand Up @@ -137,8 +126,7 @@ public struct LDUser {
}

/**
Initializer that takes a [String: Any] and creates a LDUser from the contents. Uses any keys present to define corresponding attribute values. Initializes attributes not present in the dictionary to their default value. Attempts to set `device` and `operatingSystem` from corresponding values embedded in `custom`. Attempts to set feature flags from values set in `config`.

Initializer that takes a [String: Any] and creates a LDUser from the contents. Uses any keys present to define corresponding attribute values. Initializes attributes not present in the dictionary to their default value. Attempts to set `device` and `operatingSystem` from corresponding values embedded in `custom`. DEPRECATED: Attempts to set feature flags from values set in `config`.
- parameter userDictionary: Dictionary with LDUser attribute keys and values.
*/
public init(userDictionary: [String: Any]) {
Expand Down Expand Up @@ -186,7 +174,7 @@ public struct LDUser {
case CodingKeys.custom.rawValue: return custom
case CodingKeys.device.rawValue: return device
case CodingKeys.operatingSystem.rawValue: return operatingSystem
case CodingKeys.config.rawValue: return flagStore.featureFlags
case CodingKeys.config.rawValue: return flagStore?.featureFlags
case CodingKeys.privateAttributes.rawValue: return privateAttributes
default: return nil
}
Expand Down Expand Up @@ -299,7 +287,7 @@ extension LDUserWrapper: NSCoding {
encoder.encode(wrapped.device, forKey: LDUser.CodingKeys.device.rawValue)
encoder.encode(wrapped.operatingSystem, forKey: LDUser.CodingKeys.operatingSystem.rawValue)
encoder.encode(wrapped.privateAttributes, forKey: LDUser.CodingKeys.privateAttributes.rawValue)
encoder.encode([Keys.featureFlags: wrapped.flagStore.featureFlags.dictionaryValue.withNullValuesRemoved], forKey: LDUser.CodingKeys.config.rawValue)
encoder.encode([Keys.featureFlags: wrapped.flagStore?.featureFlags.dictionaryValue.withNullValuesRemoved], forKey: LDUser.CodingKeys.config.rawValue)
}

convenience init?(coder decoder: NSCoder) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ protocol ClientServiceCreating {
func makeConnectionInformation() -> ConnectionInformation
func makeDiagnosticCache(sdkKey: String) -> DiagnosticCaching
func makeDiagnosticReporter(service: DarklyServiceProvider, runMode: LDClientRunMode) -> DiagnosticReporting
func makeFlagStore() -> FlagMaintaining
}

final class ClientServiceFactory: ClientServiceCreating {
Expand Down Expand Up @@ -142,4 +143,8 @@ final class ClientServiceFactory: ClientServiceCreating {
func makeDiagnosticReporter(service: DarklyServiceProvider, runMode: LDClientRunMode) -> DiagnosticReporting {
DiagnosticReporter(service: service, runMode: runMode)
}

func makeFlagStore() -> FlagMaintaining {
FlagStore()
}
}
Loading

0 comments on commit 89dcc5d

Please sign in to comment.