prefab-cloud-ruby 1.6.1 → 1.7.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +9 -0
- data/VERSION +1 -1
- data/lib/prefab/client.rb +1 -1
- data/lib/prefab/config_value_unwrapper.rb +2 -0
- data/lib/prefab/context.rb +23 -6
- data/lib/prefab/duration.rb +54 -0
- data/lib/prefab/logger_client.rb +2 -8
- data/lib/prefab-cloud-ruby.rb +1 -0
- data/lib/prefab_pb.rb +3 -1
- data/prefab-cloud-ruby.gemspec +5 -3
- data/test/integration_test.rb +5 -1
- data/test/integration_test_helpers.rb +5 -17
- data/test/support/common_helpers.rb +2 -0
- data/test/test_config_resolver.rb +31 -0
- data/test/test_config_value_unwrapper.rb +16 -0
- data/test/test_context.rb +87 -0
- data/test/test_context_shape.rb +1 -1
- data/test/test_duration.rb +37 -0
- data/test/test_integration.rb +2 -0
- metadata +4 -2
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: d31688d2a9cd1c87625a186f07d7c46123fe1ee298991b05debe9647414b0504
         | 
| 4 | 
            +
              data.tar.gz: 279526d7ac47e21e304c731a76a1946f164b081d79edab853d9efcbe07fc5db5
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: c13c066054f9924510ae2101ac68d0dc55561da36a696c5fd5b638e7cdb61c4efe13f7b0166cabf32ad16c638d942c9e649cd75dc03f575967c0e1e3016555ae
         | 
| 7 | 
            +
              data.tar.gz: f9c750af615228ee3852eba6d33f1d698a47e733eeec71c008be5e9b323746559734db376d8343d2630576e7d7b8be001481f8e4e27ca453b487f2ee78bf0b06
         | 
    
        data/CHANGELOG.md
    CHANGED
    
    | @@ -1,5 +1,14 @@ | |
| 1 1 | 
             
            # Changelog
         | 
| 2 2 |  | 
| 3 | 
            +
            ## 1.7.0 - 2024-04-10
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            - Add duration support (#187)
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            ## 1.6.2 - 2024-03-29
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            - Fix context telemetry when JIT and Block contexts are combined (#185)
         | 
| 10 | 
            +
            - Remove logger prefix (#186)
         | 
| 11 | 
            +
             | 
| 3 12 | 
             
            ## 1.6.1 - 2024-03-28
         | 
| 4 13 |  | 
| 5 14 | 
             
            - Performance optimizations (#178)
         | 
    
        data/VERSION
    CHANGED
    
    | @@ -1 +1 @@ | |
| 1 | 
            -
            1. | 
| 1 | 
            +
            1.7.0
         | 
    
        data/lib/prefab/client.rb
    CHANGED
    
    | @@ -14,7 +14,7 @@ module Prefab | |
| 14 14 | 
             
                  @options = options.is_a?(Prefab::Options) ? options : Prefab::Options.new(options)
         | 
| 15 15 | 
             
                  @namespace = @options.namespace
         | 
| 16 16 | 
             
                  @stubs = {}
         | 
| 17 | 
            -
                  @instance_hash = UUID.new.generate
         | 
| 17 | 
            +
                  @instance_hash = ::UUID.new.generate
         | 
| 18 18 |  | 
| 19 19 | 
             
                  if @options.local_only?
         | 
| 20 20 | 
             
                    LOG.debug 'Prefab Running in Local Mode'
         | 
| @@ -36,6 +36,8 @@ module Prefab | |
| 36 36 | 
             
                          @config_value.public_send(@config_value.type)
         | 
| 37 37 | 
             
                        when :string_list
         | 
| 38 38 | 
             
                          @config_value.string_list.values
         | 
| 39 | 
            +
                        when :duration
         | 
| 40 | 
            +
                          Prefab::Duration.new(@config_value.duration.definition)
         | 
| 39 41 | 
             
                        else
         | 
| 40 42 | 
             
                          LOG.error "Unknown type: #{@config_value.type}"
         | 
| 41 43 | 
             
                          raise "Unknown type: #{@config_value.type}"
         | 
    
        data/lib/prefab/context.rb
    CHANGED
    
    | @@ -31,7 +31,7 @@ module Prefab | |
| 31 31 | 
             
                end
         | 
| 32 32 |  | 
| 33 33 | 
             
                THREAD_KEY = :prefab_context
         | 
| 34 | 
            -
                attr_reader :contexts, :seen_at
         | 
| 34 | 
            +
                attr_reader :contexts, :seen_at, :id, :parent
         | 
| 35 35 |  | 
| 36 36 | 
             
                class << self
         | 
| 37 37 | 
             
                  def global_context=(context)
         | 
| @@ -173,6 +173,7 @@ module Prefab | |
| 173 173 |  | 
| 174 174 | 
             
                def clear
         | 
| 175 175 | 
             
                  @contexts = {}
         | 
| 176 | 
            +
                  @flattened = {}
         | 
| 176 177 | 
             
                end
         | 
| 177 178 |  | 
| 178 179 | 
             
                def context(name)
         | 
| @@ -187,6 +188,18 @@ module Prefab | |
| 187 188 | 
             
                  self
         | 
| 188 189 | 
             
                end
         | 
| 189 190 |  | 
| 191 | 
            +
                def reportable_tree
         | 
| 192 | 
            +
                  ctx = self
         | 
| 193 | 
            +
                  reportables = []
         | 
| 194 | 
            +
             | 
| 195 | 
            +
                  while ctx
         | 
| 196 | 
            +
                    reportables.unshift(ctx)
         | 
| 197 | 
            +
                    ctx = ctx.parent
         | 
| 198 | 
            +
                  end
         | 
| 199 | 
            +
             | 
| 200 | 
            +
                  reportables
         | 
| 201 | 
            +
                end
         | 
| 202 | 
            +
             | 
| 190 203 | 
             
                def to_proto(namespace)
         | 
| 191 204 | 
             
                  prefab_context = {
         | 
| 192 205 | 
             
                    'current-time' => ConfigValueWrapper.wrap(Prefab::TimeHelpers.now_in_ms)
         | 
| @@ -194,8 +207,16 @@ module Prefab | |
| 194 207 |  | 
| 195 208 | 
             
                  prefab_context['namespace'] = ConfigValueWrapper.wrap(namespace) if namespace&.length&.positive?
         | 
| 196 209 |  | 
| 210 | 
            +
                  reportable_contexts = {}
         | 
| 211 | 
            +
             | 
| 212 | 
            +
                  reportable_tree.each do |ctx|
         | 
| 213 | 
            +
                    ctx.contexts.each do |name, context|
         | 
| 214 | 
            +
                      reportable_contexts[name] = context
         | 
| 215 | 
            +
                    end
         | 
| 216 | 
            +
                  end
         | 
| 217 | 
            +
             | 
| 197 218 | 
             
                  PrefabProto::ContextSet.new(
         | 
| 198 | 
            -
                    contexts:  | 
| 219 | 
            +
                    contexts: reportable_contexts.map do |name, context|
         | 
| 199 220 | 
             
                      context.to_proto
         | 
| 200 221 | 
             
                    end.concat([PrefabProto::Context.new(type: 'prefab',
         | 
| 201 222 | 
             
                                                         values: prefab_context)])
         | 
| @@ -224,9 +245,5 @@ module Prefab | |
| 224 245 | 
             
                    super
         | 
| 225 246 | 
             
                  end
         | 
| 226 247 | 
             
                end
         | 
| 227 | 
            -
             | 
| 228 | 
            -
                def id
         | 
| 229 | 
            -
                  @id
         | 
| 230 | 
            -
                end
         | 
| 231 248 | 
             
              end
         | 
| 232 249 | 
             
            end
         | 
| @@ -0,0 +1,54 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module Prefab
         | 
| 4 | 
            +
              class Duration
         | 
| 5 | 
            +
                PATTERN = /P(?:(?<days>\d+(?:\.\d+)?)D)?(?:T(?:(?<hours>\d+(?:\.\d+)?)H)?(?:(?<minutes>\d+(?:\.\d+)?)M)?(?:(?<seconds>\d+(?:\.\d+)?)S)?)?/
         | 
| 6 | 
            +
                MINUTES_IN_SECONDS = 60
         | 
| 7 | 
            +
                HOURS_IN_SECONDS = 60 * MINUTES_IN_SECONDS
         | 
| 8 | 
            +
                DAYS_IN_SECONDS = 24 * HOURS_IN_SECONDS
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                def initialize(definition)
         | 
| 11 | 
            +
                  @seconds = self.class.parse(definition)
         | 
| 12 | 
            +
                end
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                def self.parse(definition)
         | 
| 15 | 
            +
                  match = PATTERN.match(definition)
         | 
| 16 | 
            +
                  return 0 unless match
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                  days = match[:days]&.to_f || 0
         | 
| 19 | 
            +
                  hours = match[:hours]&.to_f || 0
         | 
| 20 | 
            +
                  minutes = match[:minutes]&.to_f || 0
         | 
| 21 | 
            +
                  seconds = match[:seconds]&.to_f || 0
         | 
| 22 | 
            +
             | 
| 23 | 
            +
                  (days * DAYS_IN_SECONDS + hours * HOURS_IN_SECONDS + minutes * MINUTES_IN_SECONDS + seconds)
         | 
| 24 | 
            +
                end
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                def in_seconds
         | 
| 27 | 
            +
                  @seconds
         | 
| 28 | 
            +
                end
         | 
| 29 | 
            +
             | 
| 30 | 
            +
                def in_minutes
         | 
| 31 | 
            +
                  in_seconds / 60.0
         | 
| 32 | 
            +
                end
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                def in_hours
         | 
| 35 | 
            +
                  in_minutes / 60.0
         | 
| 36 | 
            +
                end
         | 
| 37 | 
            +
             | 
| 38 | 
            +
                def in_days
         | 
| 39 | 
            +
                  in_hours / 24.0
         | 
| 40 | 
            +
                end
         | 
| 41 | 
            +
             | 
| 42 | 
            +
                def in_weeks
         | 
| 43 | 
            +
                  in_days / 7.0
         | 
| 44 | 
            +
                end
         | 
| 45 | 
            +
             | 
| 46 | 
            +
                def to_i
         | 
| 47 | 
            +
                  in_seconds.to_i
         | 
| 48 | 
            +
                end
         | 
| 49 | 
            +
             | 
| 50 | 
            +
                def to_f
         | 
| 51 | 
            +
                  in_seconds.to_f
         | 
| 52 | 
            +
                end
         | 
| 53 | 
            +
              end
         | 
| 54 | 
            +
            end
         | 
    
        data/lib/prefab/logger_client.rb
    CHANGED
    
    | @@ -28,13 +28,11 @@ module Prefab | |
| 28 28 |  | 
| 29 29 | 
             
                def semantic_filter(log)
         | 
| 30 30 | 
             
                  class_path = class_path_name(log.name)
         | 
| 31 | 
            -
                  lookup_path = "#{logger_prefix}.#{class_path}"
         | 
| 32 31 | 
             
                  level = SemanticLogger::Levels.index(log.level)
         | 
| 33 | 
            -
                  log.named_tags.merge!({ path:  | 
| 34 | 
            -
                  should_log? level,  | 
| 32 | 
            +
                  log.named_tags.merge!({ path: class_path })
         | 
| 33 | 
            +
                  should_log? level, class_path
         | 
| 35 34 | 
             
                end
         | 
| 36 35 |  | 
| 37 | 
            -
             | 
| 38 36 | 
             
                def config_client=(config_client)
         | 
| 39 37 | 
             
                  @config_client = config_client
         | 
| 40 38 | 
             
                end
         | 
| @@ -50,10 +48,6 @@ module Prefab | |
| 50 48 | 
             
                  end.gsub(/[^a-z_]/i, '.')
         | 
| 51 49 | 
             
                end
         | 
| 52 50 |  | 
| 53 | 
            -
                def logger_prefix
         | 
| 54 | 
            -
                  Context.current.get("application.key") ||  "prefab-cloud-ruby"
         | 
| 55 | 
            -
                end
         | 
| 56 | 
            -
             | 
| 57 51 | 
             
                # Find the closest match to 'log_level.path' in config
         | 
| 58 52 | 
             
                def level_of(path)
         | 
| 59 53 | 
             
                  closest_log_level_match = nil
         | 
    
        data/lib/prefab-cloud-ruby.rb
    CHANGED
    
    
    
        data/lib/prefab_pb.rb
    CHANGED
    
    | @@ -5,7 +5,7 @@ | |
| 5 5 | 
             
            require 'google/protobuf'
         | 
| 6 6 |  | 
| 7 7 |  | 
| 8 | 
            -
            descriptor_data = "\n\x0cprefab.proto\x12\x06prefab\"W\n\x14\x43onfigServicePointer\x12\x12\n\nproject_id\x18\x01 \x01(\x03\x12\x13\n\x0bstart_at_id\x18\x02 \x01(\x03\x12\x16\n\x0eproject_env_id\x18\x03 \x01(\x03\"\xc8\x03\n\x0b\x43onfigValue\x12\r\n\x03int\x18\x01 \x01(\x03H\x00\x12\x10\n\x06string\x18\x02 \x01(\tH\x00\x12\x0f\n\x05\x62ytes\x18\x03 \x01(\x0cH\x00\x12\x10\n\x06\x64ouble\x18\x04 \x01(\x01H\x00\x12\x0e\n\x04\x62ool\x18\x05 \x01(\x08H\x00\x12\x31\n\x0fweighted_values\x18\x06 \x01(\x0b\x32\x16.prefab.WeightedValuesH\x00\x12\x33\n\x10limit_definition\x18\x07 \x01(\x0b\x32\x17.prefab.LimitDefinitionH\x00\x12%\n\tlog_level\x18\t \x01(\x0e\x32\x10.prefab.LogLevelH\x00\x12)\n\x0bstring_list\x18\n \x01(\x0b\x32\x12.prefab.StringListH\x00\x12%\n\tint_range\x18\x0b \x01(\x0b\x32\x10.prefab.IntRangeH\x00\x12$\n\x08provided\x18\x0c \x01(\x0b\x32\x10.prefab.ProvidedH\x00\x12\x19\n\x0c\x63onfidential\x18\r \x01(\x08H\x01\x88\x01\x01\x12\x19\n\x0c\x64\x65\x63rypt_with\x18\x0e \x01(\tH\x02\x88\x01\x01\x42\x06\n\x04typeB\x0f\n\r_confidentialB\x0f\n\r_decrypt_with\"b\n\x08Provided\x12+\n\x06source\x18\x01 \x01(\x0e\x32\x16.prefab.ProvidedSourceH\x00\x88\x01\x01\x12\x13\n\x06lookup\x18\x02 \x01(\tH\x01\x88\x01\x01\x42\t\n\x07_sourceB\t\n\x07_lookup\"B\n\x08IntRange\x12\x12\n\x05start\x18\x01 \x01(\x03H\x00\x88\x01\x01\x12\x10\n\x03\x65nd\x18\x02 \x01(\x03H\x01\x88\x01\x01\x42\x08\n\x06_startB\x06\n\x04_end\"\x1c\n\nStringList\x12\x0e\n\x06values\x18\x01 \x03(\t\"C\n\rWeightedValue\x12\x0e\n\x06weight\x18\x01 \x01(\x05\x12\"\n\x05value\x18\x02 \x01(\x0b\x32\x13.prefab.ConfigValue\"~\n\x0eWeightedValues\x12.\n\x0fweighted_values\x18\x01 \x03(\x0b\x32\x15.prefab.WeightedValue\x12\"\n\x15hash_by_property_name\x18\x02 \x01(\tH\x00\x88\x01\x01\x42\x18\n\x16_hash_by_property_name\"X\n\x0e\x41piKeyMetadata\x12\x13\n\x06key_id\x18\x01 \x01(\tH\x00\x88\x01\x01\x12\x14\n\x07user_id\x18\x03 \x01(\tH\x01\x88\x01\x01\x42\t\n\x07_key_idB\n\n\x08_user_idJ\x04\x08\x02\x10\x03\"\xa0\x02\n\x07\x43onfigs\x12\x1f\n\x07\x63onfigs\x18\x01 \x03(\x0b\x32\x0e.prefab.Config\x12<\n\x16\x63onfig_service_pointer\x18\x02 \x01(\x0b\x32\x1c.prefab.ConfigServicePointer\x12\x34\n\x0f\x61pikey_metadata\x18\x03 \x01(\x0b\x32\x16.prefab.ApiKeyMetadataH\x00\x88\x01\x01\x12\x30\n\x0f\x64\x65\x66\x61ult_context\x18\x04 \x01(\x0b\x32\x12.prefab.ContextSetH\x01\x88\x01\x01\x12\x17\n\nkeep_alive\x18\x05 \x01(\x08H\x02\x88\x01\x01\x42\x12\n\x10_apikey_metadataB\x12\n\x10_default_contextB\r\n\x0b_keep_alive\"\xe4\x03\n\x06\x43onfig\x12\n\n\x02id\x18\x01 \x01(\x03\x12\x12\n\nproject_id\x18\x02 \x01(\x03\x12\x0b\n\x03key\x18\x03 \x01(\t\x12%\n\nchanged_by\x18\x04 \x01(\x0b\x32\x11.prefab.ChangedBy\x12\x1f\n\x04rows\x18\x05 \x03(\x0b\x32\x11.prefab.ConfigRow\x12-\n\x10\x61llowable_values\x18\x06 \x03(\x0b\x32\x13.prefab.ConfigValue\x12\'\n\x0b\x63onfig_type\x18\x07 \x01(\x0e\x32\x12.prefab.ConfigType\x12\x15\n\x08\x64raft_id\x18\x08 \x01(\x03H\x00\x88\x01\x01\x12,\n\nvalue_type\x18\t \x01(\x0e\x32\x18.prefab.Config.ValueType\x12\x1a\n\x12send_to_client_sdk\x18\n \x01(\x08\"\x9e\x01\n\tValueType\x12\x16\n\x12NOT_SET_VALUE_TYPE\x10\x00\x12\x07\n\x03INT\x10\x01\x12\n\n\x06STRING\x10\x02\x12\t\n\x05\x42YTES\x10\x03\x12\n\n\x06\x44OUBLE\x10\x04\x12\x08\n\x04\x42OOL\x10\x05\x12\x14\n\x10LIMIT_DEFINITION\x10\x07\x12\r\n\tLOG_LEVEL\x10\t\x12\x0f\n\x0bSTRING_LIST\x10\n\x12\r\n\tINT_RANGE\x10\x0b\x42\x0b\n\t_draft_id\"?\n\tChangedBy\x12\x0f\n\x07user_id\x18\x01 \x01(\x03\x12\r\n\x05\x65mail\x18\x02 \x01(\t\x12\x12\n\napi_key_id\x18\x03 \x01(\t\"\xe4\x01\n\tConfigRow\x12\x1b\n\x0eproject_env_id\x18\x01 \x01(\x03H\x00\x88\x01\x01\x12(\n\x06values\x18\x02 \x03(\x0b\x32\x18.prefab.ConditionalValue\x12\x35\n\nproperties\x18\x03 \x03(\x0b\x32!.prefab.ConfigRow.PropertiesEntry\x1a\x46\n\x0fPropertiesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\"\n\x05value\x18\x02 \x01(\x0b\x32\x13.prefab.ConfigValue:\x02\x38\x01\x42\x11\n\x0f_project_env_id\"[\n\x10\x43onditionalValue\x12#\n\x08\x63riteria\x18\x01 \x03(\x0b\x32\x11.prefab.Criterion\x12\"\n\x05value\x18\x02 \x01(\x0b\x32\x13.prefab.ConfigValue\"\x94\x03\n\tCriterion\x12\x15\n\rproperty_name\x18\x01 \x01(\t\x12\x35\n\x08operator\x18\x02 \x01(\x0e\x32#.prefab.Criterion.CriterionOperator\x12+\n\x0evalue_to_match\x18\x03 \x01(\x0b\x32\x13.prefab.ConfigValue\"\x8b\x02\n\x11\x43riterionOperator\x12\x0b\n\x07NOT_SET\x10\x00\x12\x11\n\rLOOKUP_KEY_IN\x10\x01\x12\x15\n\x11LOOKUP_KEY_NOT_IN\x10\x02\x12\n\n\x06IN_SEG\x10\x03\x12\x0e\n\nNOT_IN_SEG\x10\x04\x12\x0f\n\x0b\x41LWAYS_TRUE\x10\x05\x12\x12\n\x0ePROP_IS_ONE_OF\x10\x06\x12\x16\n\x12PROP_IS_NOT_ONE_OF\x10\x07\x12\x19\n\x15PROP_ENDS_WITH_ONE_OF\x10\x08\x12!\n\x1dPROP_DOES_NOT_END_WITH_ONE_OF\x10\t\x12\x16\n\x12HIERARCHICAL_MATCH\x10\n\x12\x10\n\x0cIN_INT_RANGE\x10\x0b\"\x89\x01\n\x07Loggers\x12\x1f\n\x07loggers\x18\x01 \x03(\x0b\x32\x0e.prefab.Logger\x12\x10\n\x08start_at\x18\x02 \x01(\x03\x12\x0e\n\x06\x65nd_at\x18\x03 \x01(\x03\x12\x15\n\rinstance_hash\x18\x04 \x01(\t\x12\x16\n\tnamespace\x18\x05 \x01(\tH\x00\x88\x01\x01\x42\x0c\n\n_namespace\"\xd9\x01\n\x06Logger\x12\x13\n\x0blogger_name\x18\x01 \x01(\t\x12\x13\n\x06traces\x18\x02 \x01(\x03H\x00\x88\x01\x01\x12\x13\n\x06\x64\x65\x62ugs\x18\x03 \x01(\x03H\x01\x88\x01\x01\x12\x12\n\x05infos\x18\x04 \x01(\x03H\x02\x88\x01\x01\x12\x12\n\x05warns\x18\x05 \x01(\x03H\x03\x88\x01\x01\x12\x13\n\x06\x65rrors\x18\x06 \x01(\x03H\x04\x88\x01\x01\x12\x13\n\x06\x66\x61tals\x18\x07 \x01(\x03H\x05\x88\x01\x01\x42\t\n\x07_tracesB\t\n\x07_debugsB\x08\n\x06_infosB\x08\n\x06_warnsB\t\n\x07_errorsB\t\n\x07_fatals\"\x16\n\x14LoggerReportResponse\"\xdb\x03\n\rLimitResponse\x12\x0e\n\x06passed\x18\x01 \x01(\x08\x12\x12\n\nexpires_at\x18\x02 \x01(\x03\x12\x16\n\x0e\x65nforced_group\x18\x03 \x01(\t\x12\x16\n\x0e\x63urrent_bucket\x18\x04 \x01(\x03\x12\x14\n\x0cpolicy_group\x18\x05 \x01(\t\x12;\n\x0bpolicy_name\x18\x06 \x01(\x0e\x32&.prefab.LimitResponse.LimitPolicyNames\x12\x14\n\x0cpolicy_limit\x18\x07 \x01(\x05\x12\x0e\n\x06\x61mount\x18\x08 \x01(\x03\x12\x16\n\x0elimit_reset_at\x18\t \x01(\x03\x12\x39\n\x0csafety_level\x18\n \x01(\x0e\x32#.prefab.LimitDefinition.SafetyLevel\"\xa9\x01\n\x10LimitPolicyNames\x12\x0b\n\x07NOT_SET\x10\x00\x12\x14\n\x10SECONDLY_ROLLING\x10\x01\x12\x14\n\x10MINUTELY_ROLLING\x10\x03\x12\x12\n\x0eHOURLY_ROLLING\x10\x05\x12\x11\n\rDAILY_ROLLING\x10\x07\x12\x13\n\x0fMONTHLY_ROLLING\x10\x08\x12\x0c\n\x08INFINITE\x10\t\x12\x12\n\x0eYEARLY_ROLLING\x10\n\"\x99\x02\n\x0cLimitRequest\x12\x12\n\naccount_id\x18\x01 \x01(\x03\x12\x16\n\x0e\x61\x63quire_amount\x18\x02 \x01(\x05\x12\x0e\n\x06groups\x18\x03 \x03(\t\x12:\n\x0elimit_combiner\x18\x04 \x01(\x0e\x32\".prefab.LimitRequest.LimitCombiner\x12\x1e\n\x16\x61llow_partial_response\x18\x05 \x01(\x08\x12\x39\n\x0csafety_level\x18\x06 \x01(\x0e\x32#.prefab.LimitDefinition.SafetyLevel\"6\n\rLimitCombiner\x12\x0b\n\x07NOT_SET\x10\x00\x12\x0b\n\x07MINIMUM\x10\x01\x12\x0b\n\x07MAXIMUM\x10\x02\"/\n\nContextSet\x12!\n\x08\x63ontexts\x18\x01 \x03(\x0b\x32\x0f.prefab.Context\"\x96\x01\n\x07\x43ontext\x12\x11\n\x04type\x18\x01 \x01(\tH\x00\x88\x01\x01\x12+\n\x06values\x18\x02 \x03(\x0b\x32\x1b.prefab.Context.ValuesEntry\x1a\x42\n\x0bValuesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\"\n\x05value\x18\x02 \x01(\x0b\x32\x13.prefab.ConfigValue:\x02\x38\x01\x42\x07\n\x05_type\"\x93\x01\n\x08Identity\x12\x13\n\x06lookup\x18\x01 \x01(\tH\x00\x88\x01\x01\x12\x34\n\nattributes\x18\x02 \x03(\x0b\x32 .prefab.Identity.AttributesEntry\x1a\x31\n\x0f\x41ttributesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\t:\x02\x38\x01\x42\t\n\x07_lookup\"\xd6\x02\n\x18\x43onfigEvaluationMetaData\x12\x1d\n\x10\x63onfig_row_index\x18\x01 \x01(\x03H\x00\x88\x01\x01\x12$\n\x17\x63onditional_value_index\x18\x02 \x01(\x03H\x01\x88\x01\x01\x12!\n\x14weighted_value_index\x18\x03 \x01(\x03H\x02\x88\x01\x01\x12%\n\x04type\x18\x04 \x01(\x0e\x32\x12.prefab.ConfigTypeH\x03\x88\x01\x01\x12\x0f\n\x02id\x18\x05 \x01(\x03H\x04\x88\x01\x01\x12\x31\n\nvalue_type\x18\x06 \x01(\x0e\x32\x18.prefab.Config.ValueTypeH\x05\x88\x01\x01\x42\x13\n\x11_config_row_indexB\x1a\n\x18_conditional_value_indexB\x17\n\x15_weighted_value_indexB\x07\n\x05_typeB\x05\n\x03_idB\r\n\x0b_value_type\"\xc1\x02\n\x11\x43lientConfigValue\x12\r\n\x03int\x18\x01 \x01(\x03H\x00\x12\x10\n\x06string\x18\x02 \x01(\tH\x00\x12\x10\n\x06\x64ouble\x18\x03 \x01(\x01H\x00\x12\x0e\n\x04\x62ool\x18\x04 \x01(\x08H\x00\x12%\n\tlog_level\x18\x05 \x01(\x0e\x32\x10.prefab.LogLevelH\x00\x12)\n\x0bstring_list\x18\x07 \x01(\x0b\x32\x12.prefab.StringListH\x00\x12%\n\tint_range\x18\x08 \x01(\x0b\x32\x10.prefab.IntRangeH\x00\x12I\n\x1a\x63onfig_evaluation_metadata\x18\x06 \x01(\x0b\x32 .prefab.ConfigEvaluationMetaDataH\x01\x88\x01\x01\x42\x06\n\x04typeB\x1d\n\x1b_config_evaluation_metadata\"\xa4\x02\n\x11\x43onfigEvaluations\x12\x35\n\x06values\x18\x01 \x03(\x0b\x32%.prefab.ConfigEvaluations.ValuesEntry\x12\x34\n\x0f\x61pikey_metadata\x18\x02 \x01(\x0b\x32\x16.prefab.ApiKeyMetadataH\x00\x88\x01\x01\x12\x30\n\x0f\x64\x65\x66\x61ult_context\x18\x03 \x01(\x0b\x32\x12.prefab.ContextSetH\x01\x88\x01\x01\x1aH\n\x0bValuesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12(\n\x05value\x18\x02 \x01(\x0b\x32\x19.prefab.ClientConfigValue:\x02\x38\x01\x42\x12\n\x10_apikey_metadataB\x12\n\x10_default_context\"\xa8\x02\n\x0fLimitDefinition\x12;\n\x0bpolicy_name\x18\x02 \x01(\x0e\x32&.prefab.LimitResponse.LimitPolicyNames\x12\r\n\x05limit\x18\x03 \x01(\x05\x12\r\n\x05\x62urst\x18\x04 \x01(\x05\x12\x12\n\naccount_id\x18\x05 \x01(\x03\x12\x15\n\rlast_modified\x18\x06 \x01(\x03\x12\x12\n\nreturnable\x18\x07 \x01(\x08\x12\x39\n\x0csafety_level\x18\x08 \x01(\x0e\x32#.prefab.LimitDefinition.SafetyLevel\"@\n\x0bSafetyLevel\x12\x0b\n\x07NOT_SET\x10\x00\x12\x12\n\x0eL4_BEST_EFFORT\x10\x04\x12\x10\n\x0cL5_BOMBPROOF\x10\x05\"@\n\x10LimitDefinitions\x12,\n\x0b\x64\x65\x66initions\x18\x01 \x03(\x0b\x32\x17.prefab.LimitDefinition\"\x8a\x01\n\x0f\x42ufferedRequest\x12\x12\n\naccount_id\x18\x01 \x01(\x03\x12\x0e\n\x06method\x18\x02 \x01(\t\x12\x0b\n\x03uri\x18\x03 \x01(\t\x12\x0c\n\x04\x62ody\x18\x04 \x01(\t\x12\x14\n\x0climit_groups\x18\x05 \x03(\t\x12\x14\n\x0c\x63ontent_type\x18\x06 \x01(\t\x12\x0c\n\x04\x66ifo\x18\x07 \x01(\x08\"\x94\x01\n\x0c\x42\x61tchRequest\x12\x12\n\naccount_id\x18\x01 \x01(\x03\x12\x0e\n\x06method\x18\x02 \x01(\t\x12\x0b\n\x03uri\x18\x03 \x01(\t\x12\x0c\n\x04\x62ody\x18\x04 \x01(\t\x12\x14\n\x0climit_groups\x18\x05 \x03(\t\x12\x16\n\x0e\x62\x61tch_template\x18\x06 \x01(\t\x12\x17\n\x0f\x62\x61tch_separator\x18\x07 \x01(\t\" \n\rBasicResponse\x12\x0f\n\x07message\x18\x01 \x01(\t\"3\n\x10\x43reationResponse\x12\x0f\n\x07message\x18\x01 \x01(\t\x12\x0e\n\x06new_id\x18\x02 \x01(\x03\"h\n\x07IdBlock\x12\x12\n\nproject_id\x18\x01 \x01(\x03\x12\x16\n\x0eproject_env_id\x18\x02 \x01(\x03\x12\x15\n\rsequence_name\x18\x03 \x01(\t\x12\r\n\x05start\x18\x04 \x01(\x03\x12\x0b\n\x03\x65nd\x18\x05 \x01(\x03\"a\n\x0eIdBlockRequest\x12\x12\n\nproject_id\x18\x01 \x01(\x03\x12\x16\n\x0eproject_env_id\x18\x02 \x01(\x03\x12\x15\n\rsequence_name\x18\x03 \x01(\t\x12\x0c\n\x04size\x18\x04 \x01(\x03\"\x8a\x01\n\x0c\x43ontextShape\x12\x0c\n\x04name\x18\x01 \x01(\t\x12\x39\n\x0b\x66ield_types\x18\x02 \x03(\x0b\x32$.prefab.ContextShape.FieldTypesEntry\x1a\x31\n\x0f\x46ieldTypesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\x05:\x02\x38\x01\"[\n\rContextShapes\x12$\n\x06shapes\x18\x01 \x03(\x0b\x32\x14.prefab.ContextShape\x12\x16\n\tnamespace\x18\x02 \x01(\tH\x00\x88\x01\x01\x42\x0c\n\n_namespace\"C\n\rEvaluatedKeys\x12\x0c\n\x04keys\x18\x01 \x03(\t\x12\x16\n\tnamespace\x18\x02 \x01(\tH\x00\x88\x01\x01\x42\x0c\n\n_namespace\"\x93\x01\n\x0f\x45valuatedConfig\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\x16\n\x0e\x63onfig_version\x18\x02 \x01(\x03\x12#\n\x06result\x18\x03 \x01(\x0b\x32\x13.prefab.ConfigValue\x12#\n\x07\x63ontext\x18\x04 \x01(\x0b\x32\x12.prefab.ContextSet\x12\x11\n\ttimestamp\x18\x05 \x01(\x03\"<\n\x10\x45valuatedConfigs\x12(\n\x07\x63onfigs\x18\x01 \x03(\x0b\x32\x17.prefab.EvaluatedConfig\"\xc4\x03\n\x17\x43onfigEvaluationCounter\x12\r\n\x05\x63ount\x18\x01 \x01(\x03\x12\x16\n\tconfig_id\x18\x02 \x01(\x03H\x00\x88\x01\x01\x12\x1b\n\x0eselected_index\x18\x03 \x01(\rH\x01\x88\x01\x01\x12\x30\n\x0eselected_value\x18\x04 \x01(\x0b\x32\x13.prefab.ConfigValueH\x02\x88\x01\x01\x12\x1d\n\x10\x63onfig_row_index\x18\x05 \x01(\rH\x03\x88\x01\x01\x12$\n\x17\x63onditional_value_index\x18\x06 \x01(\rH\x04\x88\x01\x01\x12!\n\x14weighted_value_index\x18\x07 \x01(\rH\x05\x88\x01\x01\x12\x36\n\x06reason\x18\x08 \x01(\x0e\x32&.prefab.ConfigEvaluationCounter.Reason\"\x15\n\x06Reason\x12\x0b\n\x07UNKNOWN\x10\x00\x42\x0c\n\n_config_idB\x11\n\x0f_selected_indexB\x11\n\x0f_selected_valueB\x13\n\x11_config_row_indexB\x1a\n\x18_conditional_value_indexB\x17\n\x15_weighted_value_index\"{\n\x17\x43onfigEvaluationSummary\x12\x0b\n\x03key\x18\x01 \x01(\t\x12 \n\x04type\x18\x02 \x01(\x0e\x32\x12.prefab.ConfigType\x12\x31\n\x08\x63ounters\x18\x03 \x03(\x0b\x32\x1f.prefab.ConfigEvaluationCounter\"k\n\x19\x43onfigEvaluationSummaries\x12\r\n\x05start\x18\x01 \x01(\x03\x12\x0b\n\x03\x65nd\x18\x02 \x01(\x03\x12\x32\n\tsummaries\x18\x03 \x03(\x0b\x32\x1f.prefab.ConfigEvaluationSummary\"Z\n\x15LoggersTelemetryEvent\x12\x1f\n\x07loggers\x18\x01 \x03(\x0b\x32\x0e.prefab.Logger\x12\x10\n\x08start_at\x18\x02 \x01(\x03\x12\x0e\n\x06\x65nd_at\x18\x03 \x01(\x03\"\x98\x02\n\x0eTelemetryEvent\x12\x36\n\tsummaries\x18\x02 \x01(\x0b\x32!.prefab.ConfigEvaluationSummariesH\x00\x12\x33\n\x10\x65xample_contexts\x18\x03 \x01(\x0b\x32\x17.prefab.ExampleContextsH\x00\x12+\n\x0c\x63lient_stats\x18\x04 \x01(\x0b\x32\x13.prefab.ClientStatsH\x00\x12\x30\n\x07loggers\x18\x05 \x01(\x0b\x32\x1d.prefab.LoggersTelemetryEventH\x00\x12/\n\x0e\x63ontext_shapes\x18\x06 \x01(\x0b\x32\x15.prefab.ContextShapesH\x00\x42\t\n\x07payload\"P\n\x0fTelemetryEvents\x12\x15\n\rinstance_hash\x18\x01 \x01(\t\x12&\n\x06\x65vents\x18\x02 \x03(\x0b\x32\x16.prefab.TelemetryEvent\"*\n\x17TelemetryEventsResponse\x12\x0f\n\x07success\x18\x01 \x01(\x08\";\n\x0f\x45xampleContexts\x12(\n\x08\x65xamples\x18\x01 \x03(\x0b\x32\x16.prefab.ExampleContext\"K\n\x0e\x45xampleContext\x12\x11\n\ttimestamp\x18\x01 \x01(\x03\x12&\n\ncontextSet\x18\x02 \x01(\x0b\x32\x12.prefab.ContextSet\"F\n\x0b\x43lientStats\x12\r\n\x05start\x18\x01 \x01(\x03\x12\x0b\n\x03\x65nd\x18\x02 \x01(\x03\x12\x1b\n\x13\x64ropped_event_count\x18\x03 \x01(\x04*:\n\x0eProvidedSource\x12\x1b\n\x17PROVIDED_SOURCE_NOT_SET\x10\x00\x12\x0b\n\x07\x45NV_VAR\x10\x01*\x82\x01\n\nConfigType\x12\x17\n\x13NOT_SET_CONFIG_TYPE\x10\x00\x12\n\n\x06\x43ONFIG\x10\x01\x12\x10\n\x0c\x46\x45\x41TURE_FLAG\x10\x02\x12\r\n\tLOG_LEVEL\x10\x03\x12\x0b\n\x07SEGMENT\x10\x04\x12\x14\n\x10LIMIT_DEFINITION\x10\x05\x12\x0b\n\x07\x44\x45LETED\x10\x06*a\n\x08LogLevel\x12\x15\n\x11NOT_SET_LOG_LEVEL\x10\x00\x12\t\n\x05TRACE\x10\x01\x12\t\n\x05\x44\x45\x42UG\x10\x02\x12\x08\n\x04INFO\x10\x03\x12\x08\n\x04WARN\x10\x05\x12\t\n\x05\x45RROR\x10\x06\x12\t\n\x05\x46\x41TAL\x10\t*G\n\tOnFailure\x12\x0b\n\x07NOT_SET\x10\x00\x12\x10\n\x0cLOG_AND_PASS\x10\x01\x12\x10\n\x0cLOG_AND_FAIL\x10\x02\x12\t\n\x05THROW\x10\x03\x42\x1d\n\x13\x63loud.prefab.domainB\x06Prefabb\x06proto3"
         | 
| 8 | 
            +
            descriptor_data = "\n\x0cprefab.proto\x12\x06prefab\"W\n\x14\x43onfigServicePointer\x12\x12\n\nproject_id\x18\x01 \x01(\x03\x12\x13\n\x0bstart_at_id\x18\x02 \x01(\x03\x12\x16\n\x0eproject_env_id\x18\x03 \x01(\x03\"\xf1\x03\n\x0b\x43onfigValue\x12\r\n\x03int\x18\x01 \x01(\x03H\x00\x12\x10\n\x06string\x18\x02 \x01(\tH\x00\x12\x0f\n\x05\x62ytes\x18\x03 \x01(\x0cH\x00\x12\x10\n\x06\x64ouble\x18\x04 \x01(\x01H\x00\x12\x0e\n\x04\x62ool\x18\x05 \x01(\x08H\x00\x12\x31\n\x0fweighted_values\x18\x06 \x01(\x0b\x32\x16.prefab.WeightedValuesH\x00\x12\x33\n\x10limit_definition\x18\x07 \x01(\x0b\x32\x17.prefab.LimitDefinitionH\x00\x12%\n\tlog_level\x18\t \x01(\x0e\x32\x10.prefab.LogLevelH\x00\x12)\n\x0bstring_list\x18\n \x01(\x0b\x32\x12.prefab.StringListH\x00\x12%\n\tint_range\x18\x0b \x01(\x0b\x32\x10.prefab.IntRangeH\x00\x12$\n\x08provided\x18\x0c \x01(\x0b\x32\x10.prefab.ProvidedH\x00\x12\'\n\x08\x64uration\x18\x0f \x01(\x0b\x32\x13.prefab.IsoDurationH\x00\x12\x19\n\x0c\x63onfidential\x18\r \x01(\x08H\x01\x88\x01\x01\x12\x19\n\x0c\x64\x65\x63rypt_with\x18\x0e \x01(\tH\x02\x88\x01\x01\x42\x06\n\x04typeB\x0f\n\r_confidentialB\x0f\n\r_decrypt_with\"!\n\x0bIsoDuration\x12\x12\n\ndefinition\x18\x01 \x01(\t\"b\n\x08Provided\x12+\n\x06source\x18\x01 \x01(\x0e\x32\x16.prefab.ProvidedSourceH\x00\x88\x01\x01\x12\x13\n\x06lookup\x18\x02 \x01(\tH\x01\x88\x01\x01\x42\t\n\x07_sourceB\t\n\x07_lookup\"B\n\x08IntRange\x12\x12\n\x05start\x18\x01 \x01(\x03H\x00\x88\x01\x01\x12\x10\n\x03\x65nd\x18\x02 \x01(\x03H\x01\x88\x01\x01\x42\x08\n\x06_startB\x06\n\x04_end\"\x1c\n\nStringList\x12\x0e\n\x06values\x18\x01 \x03(\t\"C\n\rWeightedValue\x12\x0e\n\x06weight\x18\x01 \x01(\x05\x12\"\n\x05value\x18\x02 \x01(\x0b\x32\x13.prefab.ConfigValue\"~\n\x0eWeightedValues\x12.\n\x0fweighted_values\x18\x01 \x03(\x0b\x32\x15.prefab.WeightedValue\x12\"\n\x15hash_by_property_name\x18\x02 \x01(\tH\x00\x88\x01\x01\x42\x18\n\x16_hash_by_property_name\"X\n\x0e\x41piKeyMetadata\x12\x13\n\x06key_id\x18\x01 \x01(\tH\x00\x88\x01\x01\x12\x14\n\x07user_id\x18\x03 \x01(\tH\x01\x88\x01\x01\x42\t\n\x07_key_idB\n\n\x08_user_idJ\x04\x08\x02\x10\x03\"\xa0\x02\n\x07\x43onfigs\x12\x1f\n\x07\x63onfigs\x18\x01 \x03(\x0b\x32\x0e.prefab.Config\x12<\n\x16\x63onfig_service_pointer\x18\x02 \x01(\x0b\x32\x1c.prefab.ConfigServicePointer\x12\x34\n\x0f\x61pikey_metadata\x18\x03 \x01(\x0b\x32\x16.prefab.ApiKeyMetadataH\x00\x88\x01\x01\x12\x30\n\x0f\x64\x65\x66\x61ult_context\x18\x04 \x01(\x0b\x32\x12.prefab.ContextSetH\x01\x88\x01\x01\x12\x17\n\nkeep_alive\x18\x05 \x01(\x08H\x02\x88\x01\x01\x42\x12\n\x10_apikey_metadataB\x12\n\x10_default_contextB\r\n\x0b_keep_alive\"\xf2\x03\n\x06\x43onfig\x12\n\n\x02id\x18\x01 \x01(\x03\x12\x12\n\nproject_id\x18\x02 \x01(\x03\x12\x0b\n\x03key\x18\x03 \x01(\t\x12%\n\nchanged_by\x18\x04 \x01(\x0b\x32\x11.prefab.ChangedBy\x12\x1f\n\x04rows\x18\x05 \x03(\x0b\x32\x11.prefab.ConfigRow\x12-\n\x10\x61llowable_values\x18\x06 \x03(\x0b\x32\x13.prefab.ConfigValue\x12\'\n\x0b\x63onfig_type\x18\x07 \x01(\x0e\x32\x12.prefab.ConfigType\x12\x15\n\x08\x64raft_id\x18\x08 \x01(\x03H\x00\x88\x01\x01\x12,\n\nvalue_type\x18\t \x01(\x0e\x32\x18.prefab.Config.ValueType\x12\x1a\n\x12send_to_client_sdk\x18\n \x01(\x08\"\xac\x01\n\tValueType\x12\x16\n\x12NOT_SET_VALUE_TYPE\x10\x00\x12\x07\n\x03INT\x10\x01\x12\n\n\x06STRING\x10\x02\x12\t\n\x05\x42YTES\x10\x03\x12\n\n\x06\x44OUBLE\x10\x04\x12\x08\n\x04\x42OOL\x10\x05\x12\x14\n\x10LIMIT_DEFINITION\x10\x07\x12\r\n\tLOG_LEVEL\x10\t\x12\x0f\n\x0bSTRING_LIST\x10\n\x12\r\n\tINT_RANGE\x10\x0b\x12\x0c\n\x08\x44URATION\x10\x0c\x42\x0b\n\t_draft_id\"?\n\tChangedBy\x12\x0f\n\x07user_id\x18\x01 \x01(\x03\x12\r\n\x05\x65mail\x18\x02 \x01(\t\x12\x12\n\napi_key_id\x18\x03 \x01(\t\"\xe4\x01\n\tConfigRow\x12\x1b\n\x0eproject_env_id\x18\x01 \x01(\x03H\x00\x88\x01\x01\x12(\n\x06values\x18\x02 \x03(\x0b\x32\x18.prefab.ConditionalValue\x12\x35\n\nproperties\x18\x03 \x03(\x0b\x32!.prefab.ConfigRow.PropertiesEntry\x1a\x46\n\x0fPropertiesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\"\n\x05value\x18\x02 \x01(\x0b\x32\x13.prefab.ConfigValue:\x02\x38\x01\x42\x11\n\x0f_project_env_id\"[\n\x10\x43onditionalValue\x12#\n\x08\x63riteria\x18\x01 \x03(\x0b\x32\x11.prefab.Criterion\x12\"\n\x05value\x18\x02 \x01(\x0b\x32\x13.prefab.ConfigValue\"\x94\x03\n\tCriterion\x12\x15\n\rproperty_name\x18\x01 \x01(\t\x12\x35\n\x08operator\x18\x02 \x01(\x0e\x32#.prefab.Criterion.CriterionOperator\x12+\n\x0evalue_to_match\x18\x03 \x01(\x0b\x32\x13.prefab.ConfigValue\"\x8b\x02\n\x11\x43riterionOperator\x12\x0b\n\x07NOT_SET\x10\x00\x12\x11\n\rLOOKUP_KEY_IN\x10\x01\x12\x15\n\x11LOOKUP_KEY_NOT_IN\x10\x02\x12\n\n\x06IN_SEG\x10\x03\x12\x0e\n\nNOT_IN_SEG\x10\x04\x12\x0f\n\x0b\x41LWAYS_TRUE\x10\x05\x12\x12\n\x0ePROP_IS_ONE_OF\x10\x06\x12\x16\n\x12PROP_IS_NOT_ONE_OF\x10\x07\x12\x19\n\x15PROP_ENDS_WITH_ONE_OF\x10\x08\x12!\n\x1dPROP_DOES_NOT_END_WITH_ONE_OF\x10\t\x12\x16\n\x12HIERARCHICAL_MATCH\x10\n\x12\x10\n\x0cIN_INT_RANGE\x10\x0b\"\x89\x01\n\x07Loggers\x12\x1f\n\x07loggers\x18\x01 \x03(\x0b\x32\x0e.prefab.Logger\x12\x10\n\x08start_at\x18\x02 \x01(\x03\x12\x0e\n\x06\x65nd_at\x18\x03 \x01(\x03\x12\x15\n\rinstance_hash\x18\x04 \x01(\t\x12\x16\n\tnamespace\x18\x05 \x01(\tH\x00\x88\x01\x01\x42\x0c\n\n_namespace\"\xd9\x01\n\x06Logger\x12\x13\n\x0blogger_name\x18\x01 \x01(\t\x12\x13\n\x06traces\x18\x02 \x01(\x03H\x00\x88\x01\x01\x12\x13\n\x06\x64\x65\x62ugs\x18\x03 \x01(\x03H\x01\x88\x01\x01\x12\x12\n\x05infos\x18\x04 \x01(\x03H\x02\x88\x01\x01\x12\x12\n\x05warns\x18\x05 \x01(\x03H\x03\x88\x01\x01\x12\x13\n\x06\x65rrors\x18\x06 \x01(\x03H\x04\x88\x01\x01\x12\x13\n\x06\x66\x61tals\x18\x07 \x01(\x03H\x05\x88\x01\x01\x42\t\n\x07_tracesB\t\n\x07_debugsB\x08\n\x06_infosB\x08\n\x06_warnsB\t\n\x07_errorsB\t\n\x07_fatals\"\x16\n\x14LoggerReportResponse\"\xdb\x03\n\rLimitResponse\x12\x0e\n\x06passed\x18\x01 \x01(\x08\x12\x12\n\nexpires_at\x18\x02 \x01(\x03\x12\x16\n\x0e\x65nforced_group\x18\x03 \x01(\t\x12\x16\n\x0e\x63urrent_bucket\x18\x04 \x01(\x03\x12\x14\n\x0cpolicy_group\x18\x05 \x01(\t\x12;\n\x0bpolicy_name\x18\x06 \x01(\x0e\x32&.prefab.LimitResponse.LimitPolicyNames\x12\x14\n\x0cpolicy_limit\x18\x07 \x01(\x05\x12\x0e\n\x06\x61mount\x18\x08 \x01(\x03\x12\x16\n\x0elimit_reset_at\x18\t \x01(\x03\x12\x39\n\x0csafety_level\x18\n \x01(\x0e\x32#.prefab.LimitDefinition.SafetyLevel\"\xa9\x01\n\x10LimitPolicyNames\x12\x0b\n\x07NOT_SET\x10\x00\x12\x14\n\x10SECONDLY_ROLLING\x10\x01\x12\x14\n\x10MINUTELY_ROLLING\x10\x03\x12\x12\n\x0eHOURLY_ROLLING\x10\x05\x12\x11\n\rDAILY_ROLLING\x10\x07\x12\x13\n\x0fMONTHLY_ROLLING\x10\x08\x12\x0c\n\x08INFINITE\x10\t\x12\x12\n\x0eYEARLY_ROLLING\x10\n\"\x99\x02\n\x0cLimitRequest\x12\x12\n\naccount_id\x18\x01 \x01(\x03\x12\x16\n\x0e\x61\x63quire_amount\x18\x02 \x01(\x05\x12\x0e\n\x06groups\x18\x03 \x03(\t\x12:\n\x0elimit_combiner\x18\x04 \x01(\x0e\x32\".prefab.LimitRequest.LimitCombiner\x12\x1e\n\x16\x61llow_partial_response\x18\x05 \x01(\x08\x12\x39\n\x0csafety_level\x18\x06 \x01(\x0e\x32#.prefab.LimitDefinition.SafetyLevel\"6\n\rLimitCombiner\x12\x0b\n\x07NOT_SET\x10\x00\x12\x0b\n\x07MINIMUM\x10\x01\x12\x0b\n\x07MAXIMUM\x10\x02\"/\n\nContextSet\x12!\n\x08\x63ontexts\x18\x01 \x03(\x0b\x32\x0f.prefab.Context\"\x96\x01\n\x07\x43ontext\x12\x11\n\x04type\x18\x01 \x01(\tH\x00\x88\x01\x01\x12+\n\x06values\x18\x02 \x03(\x0b\x32\x1b.prefab.Context.ValuesEntry\x1a\x42\n\x0bValuesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\"\n\x05value\x18\x02 \x01(\x0b\x32\x13.prefab.ConfigValue:\x02\x38\x01\x42\x07\n\x05_type\"\x93\x01\n\x08Identity\x12\x13\n\x06lookup\x18\x01 \x01(\tH\x00\x88\x01\x01\x12\x34\n\nattributes\x18\x02 \x03(\x0b\x32 .prefab.Identity.AttributesEntry\x1a\x31\n\x0f\x41ttributesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\t:\x02\x38\x01\x42\t\n\x07_lookup\"\xd6\x02\n\x18\x43onfigEvaluationMetaData\x12\x1d\n\x10\x63onfig_row_index\x18\x01 \x01(\x03H\x00\x88\x01\x01\x12$\n\x17\x63onditional_value_index\x18\x02 \x01(\x03H\x01\x88\x01\x01\x12!\n\x14weighted_value_index\x18\x03 \x01(\x03H\x02\x88\x01\x01\x12%\n\x04type\x18\x04 \x01(\x0e\x32\x12.prefab.ConfigTypeH\x03\x88\x01\x01\x12\x0f\n\x02id\x18\x05 \x01(\x03H\x04\x88\x01\x01\x12\x31\n\nvalue_type\x18\x06 \x01(\x0e\x32\x18.prefab.Config.ValueTypeH\x05\x88\x01\x01\x42\x13\n\x11_config_row_indexB\x1a\n\x18_conditional_value_indexB\x17\n\x15_weighted_value_indexB\x07\n\x05_typeB\x05\n\x03_idB\r\n\x0b_value_type\"\xed\x02\n\x11\x43lientConfigValue\x12\r\n\x03int\x18\x01 \x01(\x03H\x00\x12\x10\n\x06string\x18\x02 \x01(\tH\x00\x12\x10\n\x06\x64ouble\x18\x03 \x01(\x01H\x00\x12\x0e\n\x04\x62ool\x18\x04 \x01(\x08H\x00\x12%\n\tlog_level\x18\x05 \x01(\x0e\x32\x10.prefab.LogLevelH\x00\x12)\n\x0bstring_list\x18\x07 \x01(\x0b\x32\x12.prefab.StringListH\x00\x12%\n\tint_range\x18\x08 \x01(\x0b\x32\x10.prefab.IntRangeH\x00\x12*\n\x08\x64uration\x18\t \x01(\x0b\x32\x16.prefab.ClientDurationH\x00\x12I\n\x1a\x63onfig_evaluation_metadata\x18\x06 \x01(\x0b\x32 .prefab.ConfigEvaluationMetaDataH\x01\x88\x01\x01\x42\x06\n\x04typeB\x1d\n\x1b_config_evaluation_metadata\"0\n\x0e\x43lientDuration\x12\x0f\n\x07seconds\x18\x01 \x01(\x03\x12\r\n\x05nanos\x18\x02 \x01(\x05\"\xa4\x02\n\x11\x43onfigEvaluations\x12\x35\n\x06values\x18\x01 \x03(\x0b\x32%.prefab.ConfigEvaluations.ValuesEntry\x12\x34\n\x0f\x61pikey_metadata\x18\x02 \x01(\x0b\x32\x16.prefab.ApiKeyMetadataH\x00\x88\x01\x01\x12\x30\n\x0f\x64\x65\x66\x61ult_context\x18\x03 \x01(\x0b\x32\x12.prefab.ContextSetH\x01\x88\x01\x01\x1aH\n\x0bValuesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12(\n\x05value\x18\x02 \x01(\x0b\x32\x19.prefab.ClientConfigValue:\x02\x38\x01\x42\x12\n\x10_apikey_metadataB\x12\n\x10_default_context\"\xa8\x02\n\x0fLimitDefinition\x12;\n\x0bpolicy_name\x18\x02 \x01(\x0e\x32&.prefab.LimitResponse.LimitPolicyNames\x12\r\n\x05limit\x18\x03 \x01(\x05\x12\r\n\x05\x62urst\x18\x04 \x01(\x05\x12\x12\n\naccount_id\x18\x05 \x01(\x03\x12\x15\n\rlast_modified\x18\x06 \x01(\x03\x12\x12\n\nreturnable\x18\x07 \x01(\x08\x12\x39\n\x0csafety_level\x18\x08 \x01(\x0e\x32#.prefab.LimitDefinition.SafetyLevel\"@\n\x0bSafetyLevel\x12\x0b\n\x07NOT_SET\x10\x00\x12\x12\n\x0eL4_BEST_EFFORT\x10\x04\x12\x10\n\x0cL5_BOMBPROOF\x10\x05\"@\n\x10LimitDefinitions\x12,\n\x0b\x64\x65\x66initions\x18\x01 \x03(\x0b\x32\x17.prefab.LimitDefinition\"\x8a\x01\n\x0f\x42ufferedRequest\x12\x12\n\naccount_id\x18\x01 \x01(\x03\x12\x0e\n\x06method\x18\x02 \x01(\t\x12\x0b\n\x03uri\x18\x03 \x01(\t\x12\x0c\n\x04\x62ody\x18\x04 \x01(\t\x12\x14\n\x0climit_groups\x18\x05 \x03(\t\x12\x14\n\x0c\x63ontent_type\x18\x06 \x01(\t\x12\x0c\n\x04\x66ifo\x18\x07 \x01(\x08\"\x94\x01\n\x0c\x42\x61tchRequest\x12\x12\n\naccount_id\x18\x01 \x01(\x03\x12\x0e\n\x06method\x18\x02 \x01(\t\x12\x0b\n\x03uri\x18\x03 \x01(\t\x12\x0c\n\x04\x62ody\x18\x04 \x01(\t\x12\x14\n\x0climit_groups\x18\x05 \x03(\t\x12\x16\n\x0e\x62\x61tch_template\x18\x06 \x01(\t\x12\x17\n\x0f\x62\x61tch_separator\x18\x07 \x01(\t\" \n\rBasicResponse\x12\x0f\n\x07message\x18\x01 \x01(\t\"3\n\x10\x43reationResponse\x12\x0f\n\x07message\x18\x01 \x01(\t\x12\x0e\n\x06new_id\x18\x02 \x01(\x03\"h\n\x07IdBlock\x12\x12\n\nproject_id\x18\x01 \x01(\x03\x12\x16\n\x0eproject_env_id\x18\x02 \x01(\x03\x12\x15\n\rsequence_name\x18\x03 \x01(\t\x12\r\n\x05start\x18\x04 \x01(\x03\x12\x0b\n\x03\x65nd\x18\x05 \x01(\x03\"a\n\x0eIdBlockRequest\x12\x12\n\nproject_id\x18\x01 \x01(\x03\x12\x16\n\x0eproject_env_id\x18\x02 \x01(\x03\x12\x15\n\rsequence_name\x18\x03 \x01(\t\x12\x0c\n\x04size\x18\x04 \x01(\x03\"\x8a\x01\n\x0c\x43ontextShape\x12\x0c\n\x04name\x18\x01 \x01(\t\x12\x39\n\x0b\x66ield_types\x18\x02 \x03(\x0b\x32$.prefab.ContextShape.FieldTypesEntry\x1a\x31\n\x0f\x46ieldTypesEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\x05:\x02\x38\x01\"[\n\rContextShapes\x12$\n\x06shapes\x18\x01 \x03(\x0b\x32\x14.prefab.ContextShape\x12\x16\n\tnamespace\x18\x02 \x01(\tH\x00\x88\x01\x01\x42\x0c\n\n_namespace\"C\n\rEvaluatedKeys\x12\x0c\n\x04keys\x18\x01 \x03(\t\x12\x16\n\tnamespace\x18\x02 \x01(\tH\x00\x88\x01\x01\x42\x0c\n\n_namespace\"\x93\x01\n\x0f\x45valuatedConfig\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\x16\n\x0e\x63onfig_version\x18\x02 \x01(\x03\x12#\n\x06result\x18\x03 \x01(\x0b\x32\x13.prefab.ConfigValue\x12#\n\x07\x63ontext\x18\x04 \x01(\x0b\x32\x12.prefab.ContextSet\x12\x11\n\ttimestamp\x18\x05 \x01(\x03\"<\n\x10\x45valuatedConfigs\x12(\n\x07\x63onfigs\x18\x01 \x03(\x0b\x32\x17.prefab.EvaluatedConfig\"\xc4\x03\n\x17\x43onfigEvaluationCounter\x12\r\n\x05\x63ount\x18\x01 \x01(\x03\x12\x16\n\tconfig_id\x18\x02 \x01(\x03H\x00\x88\x01\x01\x12\x1b\n\x0eselected_index\x18\x03 \x01(\rH\x01\x88\x01\x01\x12\x30\n\x0eselected_value\x18\x04 \x01(\x0b\x32\x13.prefab.ConfigValueH\x02\x88\x01\x01\x12\x1d\n\x10\x63onfig_row_index\x18\x05 \x01(\rH\x03\x88\x01\x01\x12$\n\x17\x63onditional_value_index\x18\x06 \x01(\rH\x04\x88\x01\x01\x12!\n\x14weighted_value_index\x18\x07 \x01(\rH\x05\x88\x01\x01\x12\x36\n\x06reason\x18\x08 \x01(\x0e\x32&.prefab.ConfigEvaluationCounter.Reason\"\x15\n\x06Reason\x12\x0b\n\x07UNKNOWN\x10\x00\x42\x0c\n\n_config_idB\x11\n\x0f_selected_indexB\x11\n\x0f_selected_valueB\x13\n\x11_config_row_indexB\x1a\n\x18_conditional_value_indexB\x17\n\x15_weighted_value_index\"{\n\x17\x43onfigEvaluationSummary\x12\x0b\n\x03key\x18\x01 \x01(\t\x12 \n\x04type\x18\x02 \x01(\x0e\x32\x12.prefab.ConfigType\x12\x31\n\x08\x63ounters\x18\x03 \x03(\x0b\x32\x1f.prefab.ConfigEvaluationCounter\"k\n\x19\x43onfigEvaluationSummaries\x12\r\n\x05start\x18\x01 \x01(\x03\x12\x0b\n\x03\x65nd\x18\x02 \x01(\x03\x12\x32\n\tsummaries\x18\x03 \x03(\x0b\x32\x1f.prefab.ConfigEvaluationSummary\"Z\n\x15LoggersTelemetryEvent\x12\x1f\n\x07loggers\x18\x01 \x03(\x0b\x32\x0e.prefab.Logger\x12\x10\n\x08start_at\x18\x02 \x01(\x03\x12\x0e\n\x06\x65nd_at\x18\x03 \x01(\x03\"\x98\x02\n\x0eTelemetryEvent\x12\x36\n\tsummaries\x18\x02 \x01(\x0b\x32!.prefab.ConfigEvaluationSummariesH\x00\x12\x33\n\x10\x65xample_contexts\x18\x03 \x01(\x0b\x32\x17.prefab.ExampleContextsH\x00\x12+\n\x0c\x63lient_stats\x18\x04 \x01(\x0b\x32\x13.prefab.ClientStatsH\x00\x12\x30\n\x07loggers\x18\x05 \x01(\x0b\x32\x1d.prefab.LoggersTelemetryEventH\x00\x12/\n\x0e\x63ontext_shapes\x18\x06 \x01(\x0b\x32\x15.prefab.ContextShapesH\x00\x42\t\n\x07payload\"P\n\x0fTelemetryEvents\x12\x15\n\rinstance_hash\x18\x01 \x01(\t\x12&\n\x06\x65vents\x18\x02 \x03(\x0b\x32\x16.prefab.TelemetryEvent\"*\n\x17TelemetryEventsResponse\x12\x0f\n\x07success\x18\x01 \x01(\x08\";\n\x0f\x45xampleContexts\x12(\n\x08\x65xamples\x18\x01 \x03(\x0b\x32\x16.prefab.ExampleContext\"K\n\x0e\x45xampleContext\x12\x11\n\ttimestamp\x18\x01 \x01(\x03\x12&\n\ncontextSet\x18\x02 \x01(\x0b\x32\x12.prefab.ContextSet\"F\n\x0b\x43lientStats\x12\r\n\x05start\x18\x01 \x01(\x03\x12\x0b\n\x03\x65nd\x18\x02 \x01(\x03\x12\x1b\n\x13\x64ropped_event_count\x18\x03 \x01(\x04*:\n\x0eProvidedSource\x12\x1b\n\x17PROVIDED_SOURCE_NOT_SET\x10\x00\x12\x0b\n\x07\x45NV_VAR\x10\x01*\x82\x01\n\nConfigType\x12\x17\n\x13NOT_SET_CONFIG_TYPE\x10\x00\x12\n\n\x06\x43ONFIG\x10\x01\x12\x10\n\x0c\x46\x45\x41TURE_FLAG\x10\x02\x12\r\n\tLOG_LEVEL\x10\x03\x12\x0b\n\x07SEGMENT\x10\x04\x12\x14\n\x10LIMIT_DEFINITION\x10\x05\x12\x0b\n\x07\x44\x45LETED\x10\x06*a\n\x08LogLevel\x12\x15\n\x11NOT_SET_LOG_LEVEL\x10\x00\x12\t\n\x05TRACE\x10\x01\x12\t\n\x05\x44\x45\x42UG\x10\x02\x12\x08\n\x04INFO\x10\x03\x12\x08\n\x04WARN\x10\x05\x12\t\n\x05\x45RROR\x10\x06\x12\t\n\x05\x46\x41TAL\x10\t*G\n\tOnFailure\x12\x0b\n\x07NOT_SET\x10\x00\x12\x10\n\x0cLOG_AND_PASS\x10\x01\x12\x10\n\x0cLOG_AND_FAIL\x10\x02\x12\t\n\x05THROW\x10\x03\x42L\n\x13\x63loud.prefab.domainB\x06PrefabZ-github.com/prefab-cloud/prefab-cloud-go/protob\x06proto3"
         | 
| 9 9 |  | 
| 10 10 | 
             
            pool = Google::Protobuf::DescriptorPool.generated_pool
         | 
| 11 11 |  | 
| @@ -34,6 +34,7 @@ end | |
| 34 34 | 
             
            module PrefabProto
         | 
| 35 35 | 
             
              ConfigServicePointer = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.ConfigServicePointer").msgclass
         | 
| 36 36 | 
             
              ConfigValue = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.ConfigValue").msgclass
         | 
| 37 | 
            +
              IsoDuration = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.IsoDuration").msgclass
         | 
| 37 38 | 
             
              Provided = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.Provided").msgclass
         | 
| 38 39 | 
             
              IntRange = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.IntRange").msgclass
         | 
| 39 40 | 
             
              StringList = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.StringList").msgclass
         | 
| @@ -60,6 +61,7 @@ module PrefabProto | |
| 60 61 | 
             
              Identity = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.Identity").msgclass
         | 
| 61 62 | 
             
              ConfigEvaluationMetaData = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.ConfigEvaluationMetaData").msgclass
         | 
| 62 63 | 
             
              ClientConfigValue = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.ClientConfigValue").msgclass
         | 
| 64 | 
            +
              ClientDuration = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.ClientDuration").msgclass
         | 
| 63 65 | 
             
              ConfigEvaluations = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.ConfigEvaluations").msgclass
         | 
| 64 66 | 
             
              LimitDefinition = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.LimitDefinition").msgclass
         | 
| 65 67 | 
             
              LimitDefinition::SafetyLevel = ::Google::Protobuf::DescriptorPool.generated_pool.lookup("prefab.LimitDefinition.SafetyLevel").enummodule
         | 
    
        data/prefab-cloud-ruby.gemspec
    CHANGED
    
    | @@ -2,16 +2,16 @@ | |
| 2 2 | 
             
            # DO NOT EDIT THIS FILE DIRECTLY
         | 
| 3 3 | 
             
            # Instead, edit Juwelier::Tasks in Rakefile, and run 'rake gemspec'
         | 
| 4 4 | 
             
            # -*- encoding: utf-8 -*-
         | 
| 5 | 
            -
            # stub: prefab-cloud-ruby 1. | 
| 5 | 
            +
            # stub: prefab-cloud-ruby 1.7.0 ruby lib
         | 
| 6 6 |  | 
| 7 7 | 
             
            Gem::Specification.new do |s|
         | 
| 8 8 | 
             
              s.name = "prefab-cloud-ruby".freeze
         | 
| 9 | 
            -
              s.version = "1. | 
| 9 | 
            +
              s.version = "1.7.0"
         | 
| 10 10 |  | 
| 11 11 | 
             
              s.required_rubygems_version = Gem::Requirement.new(">= 0".freeze) if s.respond_to? :required_rubygems_version=
         | 
| 12 12 | 
             
              s.require_paths = ["lib".freeze]
         | 
| 13 13 | 
             
              s.authors = ["Jeff Dwyer".freeze]
         | 
| 14 | 
            -
              s.date = "2024- | 
| 14 | 
            +
              s.date = "2024-04-10"
         | 
| 15 15 | 
             
              s.description = "Feature Flags, Live Config, and Dynamic Log Levels as a service".freeze
         | 
| 16 16 | 
             
              s.email = "jdwyer@prefab.cloud".freeze
         | 
| 17 17 | 
             
              s.extra_rdoc_files = [
         | 
| @@ -49,6 +49,7 @@ Gem::Specification.new do |s| | |
| 49 49 | 
             
                "lib/prefab/context_shape.rb",
         | 
| 50 50 | 
             
                "lib/prefab/context_shape_aggregator.rb",
         | 
| 51 51 | 
             
                "lib/prefab/criteria_evaluator.rb",
         | 
| 52 | 
            +
                "lib/prefab/duration.rb",
         | 
| 52 53 | 
             
                "lib/prefab/encryption.rb",
         | 
| 53 54 | 
             
                "lib/prefab/error.rb",
         | 
| 54 55 | 
             
                "lib/prefab/errors/env_var_parse_error.rb",
         | 
| @@ -96,6 +97,7 @@ Gem::Specification.new do |s| | |
| 96 97 | 
             
                "test/test_context_shape.rb",
         | 
| 97 98 | 
             
                "test/test_context_shape_aggregator.rb",
         | 
| 98 99 | 
             
                "test/test_criteria_evaluator.rb",
         | 
| 100 | 
            +
                "test/test_duration.rb",
         | 
| 99 101 | 
             
                "test/test_encryption.rb",
         | 
| 100 102 | 
             
                "test/test_evaluation_summary_aggregator.rb",
         | 
| 101 103 | 
             
                "test/test_example_contexts_aggregator.rb",
         | 
    
        data/test/integration_test.rb
    CHANGED
    
    | @@ -8,6 +8,7 @@ class IntegrationTest | |
| 8 8 | 
             
                @func = parse_function(test_data['function'])
         | 
| 9 9 | 
             
                @input = parse_input(test_data['input'])
         | 
| 10 10 | 
             
                @expected = parse_expected(test_data['expected'])
         | 
| 11 | 
            +
                @type = test_data['type']
         | 
| 11 12 | 
             
                @data = test_data['data']
         | 
| 12 13 | 
             
                @expected_data = test_data['expected_data'] || []
         | 
| 13 14 | 
             
                @aggregator = test_data['aggregator']
         | 
| @@ -18,6 +19,8 @@ class IntegrationTest | |
| 18 19 | 
             
              def test_type
         | 
| 19 20 | 
             
                if @data
         | 
| 20 21 | 
             
                  :telemetry
         | 
| 22 | 
            +
                elsif @type == "DURATION"
         | 
| 23 | 
            +
                  :duration
         | 
| 21 24 | 
             
                elsif @input[0] && @input[0].start_with?('log-level.')
         | 
| 22 25 | 
             
                  :log_level
         | 
| 23 26 | 
             
                elsif @expected[:status] == 'raise'
         | 
| @@ -88,7 +91,8 @@ class IntegrationTest | |
| 88 91 | 
             
                  status: expected['status'],
         | 
| 89 92 | 
             
                  error: parse_error_type(expected['error']),
         | 
| 90 93 | 
             
                  message: expected['message'],
         | 
| 91 | 
            -
                  value: expected['value']
         | 
| 94 | 
            +
                  value: expected['value'],
         | 
| 95 | 
            +
                  millis: expected['millis'],
         | 
| 92 96 | 
             
                }
         | 
| 93 97 | 
             
              end
         | 
| 94 98 |  | 
| @@ -5,12 +5,10 @@ module IntegrationTestHelpers | |
| 5 5 | 
             
              RAISE_IF_NO_TESTS_FOUND = ENV['PREFAB_INTEGRATION_TEST_RAISE'] == 'true'
         | 
| 6 6 |  | 
| 7 7 | 
             
              def self.find_integration_tests
         | 
| 8 | 
            -
                 | 
| 9 | 
            -
             | 
| 10 | 
            -
                files = find_versioned_test_files(version)
         | 
| 8 | 
            +
                files = find_test_files
         | 
| 11 9 |  | 
| 12 10 | 
             
                if files.none?
         | 
| 13 | 
            -
                  message = "No integration tests found | 
| 11 | 
            +
                  message = "No integration tests found"
         | 
| 14 12 | 
             
                  raise message if RAISE_IF_NO_TESTS_FOUND
         | 
| 15 13 |  | 
| 16 14 | 
             
                  puts message
         | 
| @@ -19,19 +17,9 @@ module IntegrationTestHelpers | |
| 19 17 | 
             
                files
         | 
| 20 18 | 
             
              end
         | 
| 21 19 |  | 
| 22 | 
            -
              def self. | 
| 23 | 
            -
                File. | 
| 24 | 
            -
             | 
| 25 | 
            -
                puts "No version found for integration tests: #{e.message}"
         | 
| 26 | 
            -
              end
         | 
| 27 | 
            -
             | 
| 28 | 
            -
              def self.find_versioned_test_files(version)
         | 
| 29 | 
            -
                if version.nil?
         | 
| 30 | 
            -
                  []
         | 
| 31 | 
            -
                else
         | 
| 32 | 
            -
                  Dir[File.join(SUBMODULE_PATH, "tests/#{version}/**/*")]
         | 
| 33 | 
            -
                    .select { |file| file =~ /\.ya?ml$/ }
         | 
| 34 | 
            -
                end
         | 
| 20 | 
            +
              def self.find_test_files
         | 
| 21 | 
            +
                Dir[File.join(SUBMODULE_PATH, "tests/current/**/*")]
         | 
| 22 | 
            +
                  .select { |file| file =~ /\.ya?ml$/ }
         | 
| 35 23 | 
             
              end
         | 
| 36 24 |  | 
| 37 25 | 
             
              SEVERITY_LOOKUP = Prefab::LogPathAggregator::SEVERITY_KEY.invert
         | 
| @@ -7,6 +7,8 @@ module CommonHelpers | |
| 7 7 | 
             
                $oldstderr, $stderr = $stderr, StringIO.new
         | 
| 8 8 |  | 
| 9 9 | 
             
                $logs = StringIO.new
         | 
| 10 | 
            +
                Prefab::Context.global_context.clear
         | 
| 11 | 
            +
                Prefab::Context.default_context.clear
         | 
| 10 12 | 
             
                SemanticLogger.add_appender(io: $logs, filter: Prefab.log_filter)
         | 
| 11 13 | 
             
                SemanticLogger.sync!
         | 
| 12 14 | 
             
                Timecop.freeze('2023-08-09 15:18:12 -0400')
         | 
| @@ -449,6 +449,37 @@ class TestConfigResolver < Minitest::Test | |
| 449 449 | 
             
                end
         | 
| 450 450 | 
             
              end
         | 
| 451 451 |  | 
| 452 | 
            +
              def test_context_lookup_with_no_local_context
         | 
| 453 | 
            +
                global_context = { cpu: { count: 4, speed: '2.4GHz' }, clock: { timezone: 'UTC' } }
         | 
| 454 | 
            +
                default_context = { 'prefab-api-key' => { 'user-id' => 123 } }
         | 
| 455 | 
            +
                jit_context = { user: { name: 'Frank' } }
         | 
| 456 | 
            +
             | 
| 457 | 
            +
                config = PrefabProto::Config.new( key: 'example', rows: [ PrefabProto::ConfigRow.new( values: [ PrefabProto::ConditionalValue.new( value: PrefabProto::ConfigValue.new(string: 'valueB2')) ]) ])
         | 
| 458 | 
            +
             | 
| 459 | 
            +
                client = new_client(global_context: global_context, config: [config])
         | 
| 460 | 
            +
             | 
| 461 | 
            +
                # we fake getting the default context from the API
         | 
| 462 | 
            +
                Prefab::Context.default_context = default_context
         | 
| 463 | 
            +
             | 
| 464 | 
            +
                resolver = client.resolver
         | 
| 465 | 
            +
             | 
| 466 | 
            +
                context = resolver.get("example", jit_context).context
         | 
| 467 | 
            +
             | 
| 468 | 
            +
                # This digs all the way to the global context
         | 
| 469 | 
            +
                assert_equal 4, context.get('cpu.count')
         | 
| 470 | 
            +
                assert_equal '2.4GHz', context.get('cpu.speed')
         | 
| 471 | 
            +
                assert_equal 'UTC', context.get('clock.timezone')
         | 
| 472 | 
            +
             | 
| 473 | 
            +
                # This digs to the default context
         | 
| 474 | 
            +
                assert_equal 123, context.get('prefab-api-key.user-id')
         | 
| 475 | 
            +
             | 
| 476 | 
            +
                # This uses the jit context
         | 
| 477 | 
            +
                assert_equal 'Frank', context.get('user.name')
         | 
| 478 | 
            +
             | 
| 479 | 
            +
                # This is nil in the jit context because `user` was clobbered
         | 
| 480 | 
            +
                assert_nil context.get('user.email')
         | 
| 481 | 
            +
              end
         | 
| 482 | 
            +
             | 
| 452 483 | 
             
              private
         | 
| 453 484 |  | 
| 454 485 | 
             
              def resolver_for_namespace(namespace, loader, project_env_id: TEST_ENV_ID)
         | 
| @@ -49,6 +49,22 @@ class TestConfigValueUnwrapper < Minitest::Test | |
| 49 49 | 
             
                assert_equal %w[a b c], unwrap(config_value, CONFIG, EMPTY_CONTEXT)
         | 
| 50 50 | 
             
              end
         | 
| 51 51 |  | 
| 52 | 
            +
              def test_unwrapping_duration
         | 
| 53 | 
            +
                duration = PrefabProto::IsoDuration.new(definition: "PT1.5S")
         | 
| 54 | 
            +
                config_value = PrefabProto::ConfigValue.new(duration: duration)
         | 
| 55 | 
            +
                assert_equal 1.5, unwrap(config_value, CONFIG, EMPTY_CONTEXT).in_seconds
         | 
| 56 | 
            +
                assert_equal 0.025, unwrap(config_value, CONFIG, EMPTY_CONTEXT).in_minutes
         | 
| 57 | 
            +
                assert_in_delta 0.00041667, unwrap(config_value, CONFIG, EMPTY_CONTEXT).in_hours, 0.00001
         | 
| 58 | 
            +
             | 
| 59 | 
            +
                duration = PrefabProto::IsoDuration.new(definition: "P4DT12H30M5S")
         | 
| 60 | 
            +
                config_value = PrefabProto::ConfigValue.new(duration: duration)
         | 
| 61 | 
            +
                assert_in_delta 0.6458, unwrap(config_value, CONFIG, EMPTY_CONTEXT).in_weeks, 0.0001
         | 
| 62 | 
            +
                assert_in_delta 4.521, unwrap(config_value, CONFIG, EMPTY_CONTEXT).in_days, 0.001
         | 
| 63 | 
            +
                assert_in_delta 108.5, unwrap(config_value, CONFIG, EMPTY_CONTEXT).in_hours, 0.1
         | 
| 64 | 
            +
                assert_in_delta 6510.083, unwrap(config_value, CONFIG, EMPTY_CONTEXT).in_minutes, 0.001
         | 
| 65 | 
            +
                assert_equal 390605 ,unwrap(config_value, CONFIG, EMPTY_CONTEXT).in_seconds
         | 
| 66 | 
            +
              end
         | 
| 67 | 
            +
             | 
| 52 68 | 
             
              def test_unwrapping_weighted_values
         | 
| 53 69 | 
             
                # single value
         | 
| 54 70 | 
             
                config_value = PrefabProto::ConfigValue.new(weighted_values: weighted_values([['abc', 1]]))
         | 
    
        data/test/test_context.rb
    CHANGED
    
    | @@ -154,6 +154,93 @@ class TestContext < Minitest::Test | |
| 154 154 | 
             
                ), contexts.to_proto(namespace)
         | 
| 155 155 | 
             
              end
         | 
| 156 156 |  | 
| 157 | 
            +
              def test_to_proto_with_parent
         | 
| 158 | 
            +
                global_context = { cpu: { count: 4, speed: '2.4GHz' }, clock: { timezone: 'UTC' }, magic: { key: "global-key" } }
         | 
| 159 | 
            +
                default_context = { 'prefab-api-key' => { 'user-id' => 123 } }
         | 
| 160 | 
            +
             | 
| 161 | 
            +
                Prefab::Context.global_context = global_context
         | 
| 162 | 
            +
                Prefab::Context.default_context = default_context
         | 
| 163 | 
            +
             | 
| 164 | 
            +
                Prefab::Context.current = {
         | 
| 165 | 
            +
                  user: { id: 2, email: 'parent-email' },
         | 
| 166 | 
            +
                  magic: { key: 'parent-key', rabbits: 3 },
         | 
| 167 | 
            +
                  clock: { timezone: 'PST' }
         | 
| 168 | 
            +
                }
         | 
| 169 | 
            +
             | 
| 170 | 
            +
                contexts = Prefab::Context.join(hash: {
         | 
| 171 | 
            +
                                                  user: {
         | 
| 172 | 
            +
                                                    id: 1,
         | 
| 173 | 
            +
                                                    email: 'user-email'
         | 
| 174 | 
            +
                                                  },
         | 
| 175 | 
            +
                                                  team: {
         | 
| 176 | 
            +
                                                    id: 2,
         | 
| 177 | 
            +
                                                    name: 'team-name'
         | 
| 178 | 
            +
                                                  }
         | 
| 179 | 
            +
                                                }, id: :jit, parent: Prefab::Context.current)
         | 
| 180 | 
            +
             | 
| 181 | 
            +
                expected = PrefabProto::ContextSet.new(
         | 
| 182 | 
            +
                  contexts: [
         | 
| 183 | 
            +
                    # Via global
         | 
| 184 | 
            +
                    PrefabProto::Context.new(
         | 
| 185 | 
            +
                      type: "cpu",
         | 
| 186 | 
            +
                      values: {
         | 
| 187 | 
            +
                        "count" => PrefabProto::ConfigValue.new(int: 4),
         | 
| 188 | 
            +
                        "speed" => PrefabProto::ConfigValue.new(string: "2.4GHz")
         | 
| 189 | 
            +
                      }
         | 
| 190 | 
            +
                    ),
         | 
| 191 | 
            +
                    # Via default
         | 
| 192 | 
            +
                    PrefabProto::Context.new(
         | 
| 193 | 
            +
                      type: "clock",
         | 
| 194 | 
            +
                      values: {
         | 
| 195 | 
            +
                        "timezone" => PrefabProto::ConfigValue.new(string: 'PST'),
         | 
| 196 | 
            +
                      }
         | 
| 197 | 
            +
                    ),
         | 
| 198 | 
            +
                    # via current
         | 
| 199 | 
            +
                    PrefabProto::Context.new(
         | 
| 200 | 
            +
                      type: "magic",
         | 
| 201 | 
            +
                      values: {
         | 
| 202 | 
            +
                        "key" => PrefabProto::ConfigValue.new(string: 'parent-key'),
         | 
| 203 | 
            +
                        "rabbits" => PrefabProto::ConfigValue.new(int: 3)
         | 
| 204 | 
            +
                      }
         | 
| 205 | 
            +
                    ),
         | 
| 206 | 
            +
                    # via default
         | 
| 207 | 
            +
                    PrefabProto::Context.new(
         | 
| 208 | 
            +
                      type: "prefab-api-key",
         | 
| 209 | 
            +
                      values: {
         | 
| 210 | 
            +
                        "user-id" => PrefabProto::ConfigValue.new(int: 123)
         | 
| 211 | 
            +
                      }
         | 
| 212 | 
            +
                    ),
         | 
| 213 | 
            +
                    # via jit
         | 
| 214 | 
            +
                    PrefabProto::Context.new(
         | 
| 215 | 
            +
                      type: "user",
         | 
| 216 | 
            +
                      values: {
         | 
| 217 | 
            +
                        "id" => PrefabProto::ConfigValue.new(int: 1),
         | 
| 218 | 
            +
                        "email" => PrefabProto::ConfigValue.new(string: "user-email")
         | 
| 219 | 
            +
                      }
         | 
| 220 | 
            +
                    ),
         | 
| 221 | 
            +
                    # via jit
         | 
| 222 | 
            +
                    PrefabProto::Context.new(
         | 
| 223 | 
            +
                      type: "team",
         | 
| 224 | 
            +
                      values: {
         | 
| 225 | 
            +
                        "id" => PrefabProto::ConfigValue.new(int: 2),
         | 
| 226 | 
            +
                        "name" => PrefabProto::ConfigValue.new(string: "team-name")
         | 
| 227 | 
            +
                      }
         | 
| 228 | 
            +
                    ),
         | 
| 229 | 
            +
                    # via to_proto
         | 
| 230 | 
            +
                    PrefabProto::Context.new(
         | 
| 231 | 
            +
                      type: "prefab",
         | 
| 232 | 
            +
                      values: {
         | 
| 233 | 
            +
                        'current-time' => PrefabProto::ConfigValue.new(int: Prefab::TimeHelpers.now_in_ms),
         | 
| 234 | 
            +
                      }
         | 
| 235 | 
            +
                    )
         | 
| 236 | 
            +
                  ]
         | 
| 237 | 
            +
                )
         | 
| 238 | 
            +
             | 
| 239 | 
            +
                actual = contexts.to_proto("")
         | 
| 240 | 
            +
             | 
| 241 | 
            +
                assert_equal expected, actual
         | 
| 242 | 
            +
              end
         | 
| 243 | 
            +
             | 
| 157 244 | 
             
              def test_parent_lookup
         | 
| 158 245 | 
             
                global_context = { cpu: { count: 4, speed: '2.4GHz' }, clock: { timezone: 'UTC' } }
         | 
| 159 246 | 
             
                default_context = { 'prefab-api-key' => { 'user-id' => 123 } }
         | 
    
        data/test/test_context_shape.rb
    CHANGED
    
    | @@ -36,7 +36,7 @@ class TestContextShape < Minitest::Test | |
| 36 36 |  | 
| 37 37 | 
             
              # If this test fails, it means that we've added a new type to the ConfigValue
         | 
| 38 38 | 
             
              def test_mapping_is_exhaustive
         | 
| 39 | 
            -
                unsupported = [:bytes, :limit_definition, :log_level, :weighted_values, :int_range, :provided]
         | 
| 39 | 
            +
                unsupported = [:bytes, :limit_definition, :log_level, :weighted_values, :int_range, :provided, :duration]
         | 
| 40 40 | 
             
                type_fields = PrefabProto::ConfigValue.descriptor.lookup_oneof("type").entries
         | 
| 41 41 | 
             
                supported = type_fields.entries.reject do |entry|
         | 
| 42 42 | 
             
                  unsupported.include?(entry.name.to_sym)
         | 
| @@ -0,0 +1,37 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require 'test_helper'
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            class DurationTest < Minitest::Test
         | 
| 6 | 
            +
              MINUTES_IN_SECONDS = 60
         | 
| 7 | 
            +
              HOURS_IN_SECONDS = 60 * MINUTES_IN_SECONDS
         | 
| 8 | 
            +
              DAYS_IN_SECONDS = 24 * HOURS_IN_SECONDS
         | 
| 9 | 
            +
             | 
| 10 | 
            +
              TESTS = [
         | 
| 11 | 
            +
                ['PT0M0S', 0],
         | 
| 12 | 
            +
                ['PT6M', 6 * MINUTES_IN_SECONDS],
         | 
| 13 | 
            +
                ['PT90S', 90],
         | 
| 14 | 
            +
                ['P1D', DAYS_IN_SECONDS],
         | 
| 15 | 
            +
                ['PT1.5M', 1.5 * MINUTES_IN_SECONDS],
         | 
| 16 | 
            +
                ['P0.75D', 0.75 * DAYS_IN_SECONDS],
         | 
| 17 | 
            +
                ['PT1M90.3S', MINUTES_IN_SECONDS + 90.3],
         | 
| 18 | 
            +
                ['PT1H', HOURS_IN_SECONDS],
         | 
| 19 | 
            +
                ['PT1.3H', 1.3 * HOURS_IN_SECONDS],
         | 
| 20 | 
            +
                ['P1.5DT1.5M', 1.5 * DAYS_IN_SECONDS + MINUTES_IN_SECONDS * 1.5],
         | 
| 21 | 
            +
                ['P1.5DT1.5H1.5M', 1.5 * DAYS_IN_SECONDS + 1.5 * HOURS_IN_SECONDS + MINUTES_IN_SECONDS * 1.5],
         | 
| 22 | 
            +
                ['P1.5DT1.5H1.5M3.5S', 1.5 * DAYS_IN_SECONDS + 1.5 * HOURS_IN_SECONDS + MINUTES_IN_SECONDS * 1.5 + 3.5],
         | 
| 23 | 
            +
                ['P1DT2H3M4S', DAYS_IN_SECONDS + 2 * HOURS_IN_SECONDS + 3 * MINUTES_IN_SECONDS + 4],
         | 
| 24 | 
            +
                ['PT1H30M', HOURS_IN_SECONDS + 30 * MINUTES_IN_SECONDS],
         | 
| 25 | 
            +
                ['P0.5DT0.25H', 0.5 * DAYS_IN_SECONDS + 0.25 * HOURS_IN_SECONDS],
         | 
| 26 | 
            +
                ['PT15M30S', 15 * MINUTES_IN_SECONDS + 30],
         | 
| 27 | 
            +
                ['PT0.000347222H', 0.000347222 * HOURS_IN_SECONDS],
         | 
| 28 | 
            +
                ['PT23H59M59S', 23 * HOURS_IN_SECONDS + 59 * MINUTES_IN_SECONDS + 59],
         | 
| 29 | 
            +
                ['P0.25DT3.75H', 0.25 * DAYS_IN_SECONDS + 3.75 * HOURS_IN_SECONDS],
         | 
| 30 | 
            +
              ]
         | 
| 31 | 
            +
             | 
| 32 | 
            +
              def test_parsing
         | 
| 33 | 
            +
                TESTS.each do |test|
         | 
| 34 | 
            +
                  assert_equal test[1], Prefab::Duration.parse(test[0]), "Failed parsing #{test[0]}"
         | 
| 35 | 
            +
                end
         | 
| 36 | 
            +
              end
         | 
| 37 | 
            +
            end
         | 
    
        data/test/test_integration.rb
    CHANGED
    
    | @@ -47,6 +47,8 @@ class TestIntegration < Minitest::Test | |
| 47 47 | 
             
                          expected.all? do |expected|
         | 
| 48 48 | 
             
                            assert actual.include?(expected), "#{actual} should include #{expected}"
         | 
| 49 49 | 
             
                          end
         | 
| 50 | 
            +
                        when :duration
         | 
| 51 | 
            +
                          assert_equal it.expected[:millis], it.test_client.send(it.func, *it.input).in_seconds * 1000
         | 
| 50 52 | 
             
                        else
         | 
| 51 53 | 
             
                          raise "Unknown test type: #{it.test_type}"
         | 
| 52 54 | 
             
                        end
         | 
    
        metadata
    CHANGED
    
    | @@ -1,14 +1,14 @@ | |
| 1 1 | 
             
            --- !ruby/object:Gem::Specification
         | 
| 2 2 | 
             
            name: prefab-cloud-ruby
         | 
| 3 3 | 
             
            version: !ruby/object:Gem::Version
         | 
| 4 | 
            -
              version: 1. | 
| 4 | 
            +
              version: 1.7.0
         | 
| 5 5 | 
             
            platform: ruby
         | 
| 6 6 | 
             
            authors:
         | 
| 7 7 | 
             
            - Jeff Dwyer
         | 
| 8 8 | 
             
            autorequire:
         | 
| 9 9 | 
             
            bindir: bin
         | 
| 10 10 | 
             
            cert_chain: []
         | 
| 11 | 
            -
            date: 2024- | 
| 11 | 
            +
            date: 2024-04-10 00:00:00.000000000 Z
         | 
| 12 12 | 
             
            dependencies:
         | 
| 13 13 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 14 14 | 
             
              name: concurrent-ruby
         | 
| @@ -236,6 +236,7 @@ files: | |
| 236 236 | 
             
            - lib/prefab/context_shape.rb
         | 
| 237 237 | 
             
            - lib/prefab/context_shape_aggregator.rb
         | 
| 238 238 | 
             
            - lib/prefab/criteria_evaluator.rb
         | 
| 239 | 
            +
            - lib/prefab/duration.rb
         | 
| 239 240 | 
             
            - lib/prefab/encryption.rb
         | 
| 240 241 | 
             
            - lib/prefab/error.rb
         | 
| 241 242 | 
             
            - lib/prefab/errors/env_var_parse_error.rb
         | 
| @@ -283,6 +284,7 @@ files: | |
| 283 284 | 
             
            - test/test_context_shape.rb
         | 
| 284 285 | 
             
            - test/test_context_shape_aggregator.rb
         | 
| 285 286 | 
             
            - test/test_criteria_evaluator.rb
         | 
| 287 | 
            +
            - test/test_duration.rb
         | 
| 286 288 | 
             
            - test/test_encryption.rb
         | 
| 287 289 | 
             
            - test/test_evaluation_summary_aggregator.rb
         | 
| 288 290 | 
             
            - test/test_example_contexts_aggregator.rb
         |