Class: LaunchDarkly::LDClient
- Inherits:
-
Object
- Object
- LaunchDarkly::LDClient
- Extended by:
- Forwardable
- Includes:
- Impl
- Defined in:
- lib/ldclient-rb/ldclient.rb
Overview
A client for LaunchDarkly. Client instances are thread-safe. Users should create a single client instance for the lifetime of the application.
Instance Attribute Summary collapse
-
#big_segment_store_status_provider ⇒ Object
readonly
Returns an interface for tracking the status of a Big Segment store.
-
#data_source_status_provider ⇒ LaunchDarkly::Interfaces::DataSource::StatusProvider
readonly
Returns an interface for tracking the status of the data source.
-
#data_store_status_provider ⇒ LaunchDarkly::Interfaces::DataStore::StatusProvider
readonly
Returns an interface for tracking the status of a persistent data store.
-
#flag_tracker ⇒ Object
readonly
Returns an interface for tracking changes in feature flag configurations.
Instance Method Summary collapse
-
#add_hook(hook) ⇒ Object
Add a hook to the client.
-
#all_flags_state(context, options = {}) ⇒ FeatureFlagsState
Returns a FeatureFlagsState object that encapsulates the state of all feature flags for a given context, including the flag values and also metadata that can be used on the front end.
-
#close ⇒ void
Releases all network connections and other resources held by the client, making it no longer usable.
-
#flush ⇒ Object
Tells the client that all pending analytics events should be delivered as soon as possible.
-
#identify(context) ⇒ void
Registers the context.
-
#initialize(sdk_key, config = Config.default, wait_for_sec = 5) ⇒ LDClient
constructor
Creates a new client instance that connects to LaunchDarkly.
-
#initialized? ⇒ Boolean
Returns whether the client has been initialized and is ready to serve feature flag requests.
-
#migration_variation(key, context, default_stage) ⇒ Array<Symbol, Interfaces::Migrations::OpTracker>
This method returns the migration stage of the migration feature flag for the given evaluation context.
-
#secure_mode_hash(context) ⇒ String?
Creates a hash string that can be used by the JavaScript SDK to identify a context.
-
#track(event_name, context, data = nil, metric_value = nil) ⇒ void
Tracks that a context performed an event.
-
#track_migration_op(tracker) ⇒ Object
Tracks the results of a migrations operation.
-
#variation(key, context, default) ⇒ Object
Determines the variation of a feature flag to present for a context.
-
#variation_detail(key, context, default) ⇒ EvaluationDetail
Determines the variation of a feature flag for a context, like #variation, but also provides additional information about how this value was calculated.
Constructor Details
#initialize(sdk_key, config = Config.default, wait_for_sec = 5) ⇒ LDClient
Creates a new client instance that connects to LaunchDarkly. A custom configuration parameter can also supplied to specify advanced options, but for most use cases, the default configuration is appropriate.
The client will immediately attempt to connect to LaunchDarkly and retrieve your feature flag data. If it cannot successfully do so within the time limit specified by ‘wait_for_sec`, the constructor will return a client that is in an uninitialized state. See #initialized? for more details.
47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 |
# File 'lib/ldclient-rb/ldclient.rb', line 47 def initialize(sdk_key, config = Config.default, wait_for_sec = 5) # Note that sdk_key is normally a required parameter, and a nil value would cause the SDK to # fail in most configurations. However, there are some configurations where it would be OK # (offline = true, *or* we are using LDD mode or the file data source and events are disabled # so we're not connecting to any LD services) so rather than try to check for all of those # up front, we will let the constructors for the data source implementations implement this # fail-fast as appropriate, and just check here for the part regarding events. if !config.offline? && config.send_events raise ArgumentError, "sdk_key must not be nil" if sdk_key.nil? end @sdk_key = sdk_key @hooks = Concurrent::Array.new(config.hooks) @shared_executor = Concurrent::SingleThreadExecutor.new data_store_broadcaster = LaunchDarkly::Impl::Broadcaster.new(@shared_executor, config.logger) store_sink = LaunchDarkly::Impl::DataStore::UpdateSink.new(data_store_broadcaster) # We need to wrap the feature store object with a FeatureStoreClientWrapper in order to add # some necessary logic around updates. Unfortunately, we have code elsewhere that accesses # the feature store through the Config object, so we need to make a new Config that uses # the wrapped store. @store = Impl::FeatureStoreClientWrapper.new(config.feature_store, store_sink, config.logger) updated_config = config.clone updated_config.instance_variable_set(:@feature_store, @store) @config = updated_config @data_store_status_provider = LaunchDarkly::Impl::DataStore::StatusProvider.new(@store, store_sink) @big_segment_store_manager = Impl::BigSegmentStoreManager.new(config.big_segments, @config.logger) @big_segment_store_status_provider = @big_segment_store_manager.status_provider get_flag = lambda { |key| @store.get(FEATURES, key) } get_segment = lambda { |key| @store.get(SEGMENTS, key) } get_big_segments_membership = lambda { |key| @big_segment_store_manager.get_context_membership(key) } @evaluator = LaunchDarkly::Impl::Evaluator.new(get_flag, get_segment, get_big_segments_membership, @config.logger) if !@config.offline? && @config.send_events && !@config.diagnostic_opt_out? diagnostic_accumulator = Impl::DiagnosticAccumulator.new(Impl::DiagnosticAccumulator.create_diagnostic_id(sdk_key)) else diagnostic_accumulator = nil end if @config.offline? || !@config.send_events @event_processor = NullEventProcessor.new else @event_processor = EventProcessor.new(sdk_key, config, nil, diagnostic_accumulator) end if @config.use_ldd? @config.logger.info { "[LDClient] Started LaunchDarkly Client in LDD mode" } @data_source = NullUpdateProcessor.new return # requestor and update processor are not used in this mode end flag_tracker_broadcaster = LaunchDarkly::Impl::Broadcaster.new(@shared_executor, @config.logger) @flag_tracker = LaunchDarkly::Impl::FlagTracker.new(flag_tracker_broadcaster, lambda { |key, context| variation(key, context, nil) }) data_source_broadcaster = LaunchDarkly::Impl::Broadcaster.new(@shared_executor, @config.logger) # Make the update sink available on the config so that our data source factory can access the sink with a shared executor. @config.data_source_update_sink = LaunchDarkly::Impl::DataSource::UpdateSink.new(@store, data_source_broadcaster, flag_tracker_broadcaster) @data_source_status_provider = LaunchDarkly::Impl::DataSource::StatusProvider.new(data_source_broadcaster, @config.data_source_update_sink) data_source_or_factory = @config.data_source || self.method(:create_default_data_source) if data_source_or_factory.respond_to? :call # Currently, data source factories take two parameters unless they need to be aware of diagnostic_accumulator, in # which case they take three parameters. This will be changed in the future to use a less awkware mechanism. if data_source_or_factory.arity == 3 @data_source = data_source_or_factory.call(sdk_key, @config, diagnostic_accumulator) else @data_source = data_source_or_factory.call(sdk_key, @config) end else @data_source = data_source_or_factory end ready = @data_source.start return unless wait_for_sec > 0 if wait_for_sec > 60 @config.logger.warn { "[LDClient] Client was configured to block for up to #{wait_for_sec} seconds when initializing. We recommend blocking no longer than 60." } end ok = ready.wait(wait_for_sec) if !ok @config.logger.error { "[LDClient] Timeout encountered waiting for LaunchDarkly client initialization" } elsif !@data_source.initialized? @config.logger.error { "[LDClient] LaunchDarkly client initialization failed" } end end |
Instance Attribute Details
#big_segment_store_status_provider ⇒ Object (readonly)
Returns an interface for tracking the status of a Big Segment store.
The Interfaces::BigSegmentStoreStatusProvider has methods for checking whether the Big Segment store is (as far as the SDK knows) currently operational and tracking changes in this status.
594 595 596 |
# File 'lib/ldclient-rb/ldclient.rb', line 594 def big_segment_store_status_provider @big_segment_store_status_provider end |
#data_source_status_provider ⇒ LaunchDarkly::Interfaces::DataSource::StatusProvider (readonly)
Returns an interface for tracking the status of the data source.
The data source is the mechanism that the SDK uses to get feature flag configurations, such as a streaming connection (the default) or poll requests. The Interfaces::DataSource::StatusProvider has methods for checking whether the data source is (as far as the SDK knows) currently operational and tracking changes in this status.
621 622 623 |
# File 'lib/ldclient-rb/ldclient.rb', line 621 def data_source_status_provider @data_source_status_provider end |
#data_store_status_provider ⇒ LaunchDarkly::Interfaces::DataStore::StatusProvider (readonly)
Returns an interface for tracking the status of a persistent data store.
The Interfaces::DataStore::StatusProvider has methods for checking whether the data store is (as far as the SDK knows) currently operational, tracking changes in this status, and getting cache statistics. These are only relevant for a persistent data store; if you are using an in-memory data store, then this method will return a stub object that provides no information.
608 609 610 |
# File 'lib/ldclient-rb/ldclient.rb', line 608 def data_store_status_provider @data_store_status_provider end |
#flag_tracker ⇒ Object (readonly)
Returns an interface for tracking changes in feature flag configurations.
The Interfaces::FlagTracker contains methods for requesting notifications about feature flag changes using an event listener model.
630 631 632 |
# File 'lib/ldclient-rb/ldclient.rb', line 630 def flag_tracker @flag_tracker end |
Instance Method Details
#add_hook(hook) ⇒ Object
Add a hook to the client. In order to register a hook before the client starts, please use the ‘hooks` property of #LDConfig.
Hooks provide entrypoints which allow for observation of SDK functions.
150 151 152 153 154 155 156 157 |
# File 'lib/ldclient-rb/ldclient.rb', line 150 def add_hook(hook) unless hook.is_a?(Interfaces::Hooks::Hook) @config.logger.error { "[LDClient] Attempted to add a hook that does not include the LaunchDarkly::Intefaces::Hooks::Hook mixin. Ignoring." } return end @hooks.push(hook) end |
#all_flags_state(context, options = {}) ⇒ FeatureFlagsState
Returns a FeatureFlagsState object that encapsulates the state of all feature flags for a given context, including the flag values and also metadata that can be used on the front end. This method does not send analytics events back to LaunchDarkly.
515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 |
# File 'lib/ldclient-rb/ldclient.rb', line 515 def all_flags_state(context, ={}) return FeatureFlagsState.new(false) if @config.offline? unless initialized? if @store.initialized? @config.logger.warn { "Called all_flags_state before client initialization; using last known values from data store" } else @config.logger.warn { "Called all_flags_state before client initialization. Data store not available; returning empty state" } return FeatureFlagsState.new(false) end end context = Impl::Context::make_context(context) unless context.valid? @config.logger.error { "[LDClient] Context was invalid for all_flags_state (#{context.error})" } return FeatureFlagsState.new(false) end begin features = @store.all(FEATURES) rescue => exn Util.log_exception(@config.logger, "Unable to read flags for all_flags_state", exn) return FeatureFlagsState.new(false) end state = FeatureFlagsState.new(true) client_only = [:client_side_only] || false with_reasons = [:with_reasons] || false details_only_if_tracked = [:details_only_for_tracked_flags] || false features.each do |k, f| if client_only && !f[:clientSide] next end begin (eval_result, eval_state) = @evaluator.evaluate(f, context) detail = eval_result.detail rescue => exn detail = EvaluationDetail.new(nil, nil, EvaluationReason::error(EvaluationReason::ERROR_EXCEPTION)) Util.log_exception(@config.logger, "Error evaluating flag \"#{k}\" in all_flags_state", exn) end requires_experiment_data = experiment?(f, detail.reason) flag_state = { key: f[:key], value: detail.value, variation: detail.variation_index, reason: detail.reason, prerequisites: eval_state.prerequisites, version: f[:version], trackEvents: f[:trackEvents] || requires_experiment_data, trackReason: requires_experiment_data, debugEventsUntilDate: f[:debugEventsUntilDate], } state.add_flag(flag_state, with_reasons, details_only_if_tracked) end state end |
#close ⇒ void
This method returns an undefined value.
Releases all network connections and other resources held by the client, making it no longer usable.
579 580 581 582 583 584 585 586 |
# File 'lib/ldclient-rb/ldclient.rb', line 579 def close @config.logger.info { "[LDClient] Closing LaunchDarkly client..." } @data_source.stop @event_processor.stop @big_segment_store_manager.stop @store.stop @shared_executor.shutdown end |
#flush ⇒ Object
Tells the client that all pending analytics events should be delivered as soon as possible.
When the LaunchDarkly client generates analytics events (from #variation, #variation_detail, #identify, or #track), they are queued on a worker thread. The event thread normally sends all queued events to LaunchDarkly at regular intervals, controlled by the Config#flush_interval option. Calling ‘flush` triggers a send without waiting for the next interval.
Flushing is asynchronous, so this method will return before it is complete. However, if you call #close, events are guaranteed to be sent before that method returns.
171 172 173 |
# File 'lib/ldclient-rb/ldclient.rb', line 171 def flush @event_processor.flush end |
#identify(context) ⇒ void
This method returns an undefined value.
Registers the context. This method simply creates an analytics event containing the context properties, so that LaunchDarkly will know about that context if it does not already.
Calling #variation or #variation_detail also sends the context information to LaunchDarkly (if events are enabled), so you only need to use #identify if you want to identify the context without evaluating a flag.
Note that event delivery is asynchronous, so the event may not actually be sent until later; see #flush.
426 427 428 429 430 431 432 433 434 435 436 437 438 439 |
# File 'lib/ldclient-rb/ldclient.rb', line 426 def identify(context) context = LaunchDarkly::Impl::Context.make_context(context) unless context.valid? @config.logger.warn("Identify called with invalid context: #{context.error}") return end if context.key == "" @config.logger.warn("Identify called with empty key") return end @event_processor.record_identify_event(context) end |
#initialized? ⇒ Boolean
Returns whether the client has been initialized and is ready to serve feature flag requests.
If this returns false, it means that the client did not succeed in connecting to LaunchDarkly within the time limit that you specified in the constructor. It could still succeed in connecting at a later time (on another thread), or it could have given up permanently (for instance, if your SDK key is invalid). In the meantime, any call to #variation or #variation_detail will behave as follows:
-
It will check whether the feature store already contains data (that is, you
are using a database-backed store and it was populated by a previous run of this application). If so, it will use the last known feature flag data.
-
Failing that, it will return the value that you specified for the ‘default`
parameter of #variation or #variation_detail.
210 211 212 |
# File 'lib/ldclient-rb/ldclient.rb', line 210 def initialized? @config.offline? || @config.use_ldd? || @data_source.initialized? end |
#migration_variation(key, context, default_stage) ⇒ Array<Symbol, Interfaces::Migrations::OpTracker>
This method returns the migration stage of the migration feature flag for the given evaluation context.
This method returns the default stage if there is an error or the flag does not exist. If the default stage is not a valid stage, then a default stage of ‘off’ will be used instead.
385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 |
# File 'lib/ldclient-rb/ldclient.rb', line 385 def migration_variation(key, context, default_stage) unless Migrations::VALID_STAGES.include? default_stage @config.logger.error { "[LDClient] default_stage #{default_stage} is not a valid stage; continuing with 'off' as default" } default_stage = Migrations::STAGE_OFF end context = Impl::Context::make_context(context) result = evaluate_with_hooks(key, context, default_stage, :migration_variation) do detail, flag, _ = variation_with_flag(key, context, default_stage.to_s) stage = detail.value stage = stage.to_sym if stage.respond_to? :to_sym if Migrations::VALID_STAGES.include?(stage) tracker = Impl::Migrations::OpTracker.new(@config.logger, key, flag, context, detail, default_stage) next LaunchDarkly::Impl::EvaluationWithHookResult.new(detail, {stage: stage, tracker: tracker}) end detail = LaunchDarkly::Impl::Evaluator.error_result(LaunchDarkly::EvaluationReason::ERROR_WRONG_TYPE, default_stage.to_s) tracker = Impl::Migrations::OpTracker.new(@config.logger, key, flag, context, detail, default_stage) LaunchDarkly::Impl::EvaluationWithHookResult.new(detail, {stage: default_stage, tracker: tracker}) end [result.results[:stage], result.results[:tracker]] end |
#secure_mode_hash(context) ⇒ String?
Creates a hash string that can be used by the JavaScript SDK to identify a context. For more information, see [Secure mode](docs.launchdarkly.com/sdk/features/secure-mode#ruby).
182 183 184 185 186 187 188 189 190 |
# File 'lib/ldclient-rb/ldclient.rb', line 182 def secure_mode_hash(context) context = Impl::Context.make_context(context) unless context.valid? @config.logger.warn("secure_mode_hash called with invalid context: #{context.error}") return nil end OpenSSL::HMAC.hexdigest("sha256", @sdk_key, context.fully_qualified_key) end |
#track(event_name, context, data = nil, metric_value = nil) ⇒ void
This method returns an undefined value.
Tracks that a context performed an event. This method creates a “custom” analytics event containing the specified event name (key), context properties, and optional data.
Note that event delivery is asynchronous, so the event may not actually be sent until later; see #flush.
As of this version’s release date, the LaunchDarkly service does not support the ‘metricValue` parameter. As a result, specifying `metricValue` will not yet produce any different behavior from omitting it. Refer to the [SDK reference guide](docs.launchdarkly.com/sdk/features/events#ruby) for the latest status.
462 463 464 465 466 467 468 469 470 |
# File 'lib/ldclient-rb/ldclient.rb', line 462 def track(event_name, context, data = nil, metric_value = nil) context = LaunchDarkly::Impl::Context.make_context(context) unless context.valid? @config.logger.warn("Track called with invalid context: #{context.error}") return end @event_processor.record_custom_event(context, event_name, data, metric_value) end |
#track_migration_op(tracker) ⇒ Object
Tracks the results of a migrations operation. This event includes measurements which can be used to enhance the observability of a migration within the LaunchDarkly UI.
This event should be generated through Interfaces::Migrations::OpTracker. If you are using the Interfaces::Migrations::Migrator to handle migrations, this event will be created and emitted automatically.
482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 |
# File 'lib/ldclient-rb/ldclient.rb', line 482 def track_migration_op(tracker) unless tracker.is_a? LaunchDarkly::Interfaces::Migrations::OpTracker @config.logger.error { "invalid op tracker received in track_migration_op" } return end event = tracker.build if event.is_a? String @config.logger.error { "[LDClient] Error occurred generating migration op event; #{event}" } return end @event_processor.record_migration_op_event(event) end |
#variation(key, context, default) ⇒ Object
Determines the variation of a feature flag to present for a context.
225 226 227 228 229 230 231 232 233 |
# File 'lib/ldclient-rb/ldclient.rb', line 225 def variation(key, context, default) context = Impl::Context::make_context(context) result = evaluate_with_hooks(key, context, default, :variation) do detail, _, _ = variation_with_flag(key, context, default) LaunchDarkly::Impl::EvaluationWithHookResult.new(detail) end result.evaluation_detail.value end |
#variation_detail(key, context, default) ⇒ EvaluationDetail
Determines the variation of a feature flag for a context, like #variation, but also provides additional information about how this value was calculated.
The return value of ‘variation_detail` is an EvaluationDetail object, which has three properties: the result value, the positional index of this value in the flag’s list of variations, and an object describing the main reason why this value was selected. See EvaluationDetail for more on these properties.
Calling ‘variation_detail` instead of `variation` also causes the “reason” data to be included in analytics events, if you are capturing detailed event data for this flag.
For more information, see the reference guide on [Evaluation reasons](docs.launchdarkly.com/sdk/concepts/evaluation-reasons).
258 259 260 261 262 263 264 265 266 |
# File 'lib/ldclient-rb/ldclient.rb', line 258 def variation_detail(key, context, default) context = Impl::Context::make_context(context) result = evaluate_with_hooks(key, context, default, :variation_detail) do detail, _, _ = evaluate_internal(key, context, default, true) LaunchDarkly::Impl::EvaluationWithHookResult.new(detail) end result.evaluation_detail end |