shared_broker 1.5.0 → 1.6.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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 7fef9571ebd7987fe360d47b8a4374206808487afc3b641b19cb19c6dbfaf85f
4
- data.tar.gz: 98fc3d5712273e6d426ac380e5ebd3ff4c9cd6319a1eb00b333ea25cc977cb0f
3
+ metadata.gz: 24d5ac30bc2bef8c5fd99edbcb8e7e986fa8005fde7cfdb39bfb4f3c6010efa8
4
+ data.tar.gz: d66d983238edb492bddf145c48866ce83a55b88bf6806472910bdafdbbd418ec
5
5
  SHA512:
6
- metadata.gz: b6d52305cbfe8cd4e72d9e5bcbd84cc8bbda1c5c738a43e2727586e82f88bbe01d9e060d31ee28443c0b73172b4206ba33fd8640be394e6fba6b9b190f808a44
7
- data.tar.gz: fdf993c33a31f8364aa613123159aebbeeae8ebfdd4f5548d6f1eb7210b318e0cc81f660e7ab28d4bc998868386c63086fbcc793d0cfe3ade14ffae82f8afd58
6
+ metadata.gz: c25d32bc67d7c28427c533b1f3e23b048a7513740f0e8a51e067f5f5d861b761a3189b7d84d06b977400b21bf022c003c50d4515e46cfdc6cee064d36acbdb38
7
+ data.tar.gz: 1ab4165d6750a4f1d28f3ce67c070d58135d10c793b4eac944ba257b8b01f3dcd0665e36018aa0c32b27076439297e6e8ee0b789f5b943fa2820ef9472d2f09f
data/CHANGELOG.md CHANGED
@@ -5,6 +5,14 @@ All notable changes to this project will be documented in this file.
5
5
  The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
6
6
  and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
7
7
 
8
+ ## [1.6.0] - 2026-06-16
9
+
10
+ ### Added
11
+ - **Encryption Key Rotation & Granularity**: Introduced flexible topic-based encryption keys and multi-version key rotation.
12
+ - `SharedBroker::KeyProvider::Registry` to register multiple keys by key ID and route active keys based on topic glob patterns.
13
+ - `SharedBroker::KeyProvider::Static` for seamless backward compatibility with single global encryption keys.
14
+ - Automatic versioning of encrypted payloads with `_key_id` metadata, enabling decrypting historical messages using rotated keys.
15
+
8
16
  ## [1.5.0] - 2026-06-10
9
17
 
10
18
  ### Added
data/README.md CHANGED
@@ -160,6 +160,41 @@ HYBRID_BROKER = SharedBroker::Client.new(
160
160
  )
161
161
  ```
162
162
 
163
+ #### E. Encryption Key Rotation & Granularity
164
+
165
+ If your system requires encrypting payloads with different keys based on the topic (e.g., highly sensitive financial data vs. general notifications) or rotating keys without breaking the decryption of historical messages in queues, you can configure a Key Provider Registry:
166
+
167
+ ```ruby
168
+ # 1. Initialize Registry with a map of historical/current keys and active key mappings
169
+ key_registry = SharedBroker::KeyProvider::Registry.new(
170
+ keys: {
171
+ "v1" => "a" * 32, # historical key
172
+ "v2" => "b" * 32, # current general key
173
+ "finance_key_1" => "c" * 32 # current finance key
174
+ },
175
+ active_keys: {
176
+ # Topic-specific key mapping using glob patterns
177
+ "payment.*" => "finance_key_1",
178
+ # Fallback key mapping for all other topics
179
+ "*" => "v2"
180
+ }
181
+ )
182
+
183
+ # 2. Register key provider globally or pass it to Client initialize
184
+ SharedBroker.key_provider = key_registry
185
+
186
+ # Alternatively, pass it directly to the client
187
+ SPOT_BROKER = SharedBroker::Client.new(
188
+ adapter: BROKER_ADAPTER,
189
+ key_provider: key_registry
190
+ )
191
+ ```
192
+
193
+ With a key provider registry configured:
194
+ - **Publishing**: Payloads are encrypted using the active key matching the topic pattern, and a `_key_id` metadata tag is automatically appended to the envelope.
195
+ - **Subscribing**: The gem automatically reads the `_key_id` from the payload envelope and decrypts it using the correct historical key version.
196
+ - **Fallback**: If no `_key_id` is present on a received message (e.g., legacy message), it falls back to the key associated with the topic pattern.
197
+
163
198
  ---
164
199
 
165
200
  ## Usage
@@ -1,69 +1,98 @@
1
- # frozen_string_literal: true
2
-
3
- require "openssl"
4
- require "base64"
5
- require "json"
6
-
7
- module SharedBroker
8
- module Cipher
9
- class DecryptionError < StandardError; end
10
-
11
- ALGORITHM = "aes-256-gcm"
12
-
13
- def self.encrypt(payload_hash, key)
14
- return payload_hash unless key
15
-
16
- unless payload_hash.is_a?(Hash)
17
- raise ArgumentError, "Expected payload_hash to be a Hash, got #{payload_hash.class} with value #{payload_hash.inspect}"
18
- end
19
-
20
- # We don't want to encrypt correlation ID or special metadata keys if we want the broker to read them,
21
- # but we want to encrypt the actual payload data.
22
- # Let's extract metadata starting with underscore and encrypt the rest.
23
- metadata = payload_hash.select { |k, _| k.to_s.start_with?("_") }
24
- data_to_encrypt = payload_hash.reject { |k, _| k.to_s.start_with?("_") }
25
-
26
- cipher = OpenSSL::Cipher.new(ALGORITHM)
27
- cipher.encrypt
28
- cipher.key = key
29
- iv = cipher.random_iv
30
-
31
- encrypted_data = cipher.update(data_to_encrypt.to_json) + cipher.final
32
- auth_tag = cipher.auth_tag
33
-
34
- metadata.merge(
35
- _encrypted: true,
36
- _iv: Base64.strict_encode64(iv),
37
- _auth_tag: Base64.strict_encode64(auth_tag),
38
- _data: Base64.strict_encode64(encrypted_data)
39
- )
40
- end
41
-
42
- def self.decrypt(payload_hash, key)
43
- return payload_hash unless payload_hash.is_a?(Hash) && payload_hash[:_encrypted]
44
- return payload_hash unless key
45
-
46
- cipher = OpenSSL::Cipher.new(ALGORITHM)
47
- cipher.decrypt
48
- cipher.key = key
49
- cipher.iv = Base64.strict_decode64(payload_hash[:_iv])
50
- cipher.auth_tag = Base64.strict_decode64(payload_hash[:_auth_tag])
51
-
52
- encrypted_bytes = Base64.strict_decode64(payload_hash[:_data])
53
- decrypted_json = cipher.update(encrypted_bytes) + cipher.final
54
-
55
- decrypted_data = JSON.parse(decrypted_json, symbolize_names: true)
56
-
57
- # Merge back the unencrypted metadata
58
- metadata = payload_hash.select { |k, _| k.to_s.start_with?("_") }
59
- metadata.delete(:_encrypted)
60
- metadata.delete(:_iv)
61
- metadata.delete(:_auth_tag)
62
- metadata.delete(:_data)
63
-
64
- decrypted_data.merge(metadata)
65
- rescue => e
66
- raise DecryptionError, "Failed to decrypt payload. Error: #{e.message}. Offending payload: #{payload_hash.inspect}"
67
- end
68
- end
69
- end
1
+ # frozen_string_literal: true
2
+
3
+ require "openssl"
4
+ require "base64"
5
+ require "json"
6
+
7
+ module SharedBroker
8
+ module Cipher
9
+ class DecryptionError < StandardError; end
10
+
11
+ ALGORITHM = "aes-256-gcm"
12
+
13
+ def self.encrypt(payload_hash, key_provider_or_key, topic: nil)
14
+ return payload_hash unless key_provider_or_key
15
+
16
+ unless payload_hash.is_a?(Hash)
17
+ raise ArgumentError, "Expected payload_hash to be a Hash, got #{payload_hash.class} with value #{payload_hash.inspect}"
18
+ end
19
+
20
+ key, key_id = resolve_encryption_key(key_provider_or_key, topic)
21
+ return payload_hash unless key
22
+
23
+ metadata = payload_hash.select { |k, _| k.to_s.start_with?("_") }
24
+ data_to_encrypt = payload_hash.reject { |k, _| k.to_s.start_with?("_") }
25
+
26
+ cipher = OpenSSL::Cipher.new(ALGORITHM)
27
+ cipher.encrypt
28
+ cipher.key = key
29
+ iv = cipher.random_iv
30
+
31
+ encrypted_data = cipher.update(data_to_encrypt.to_json) + cipher.final
32
+ auth_tag = cipher.auth_tag
33
+
34
+ envelope = {
35
+ _encrypted: true,
36
+ _iv: Base64.strict_encode64(iv),
37
+ _auth_tag: Base64.strict_encode64(auth_tag),
38
+ _data: Base64.strict_encode64(encrypted_data)
39
+ }
40
+ envelope[:_key_id] = key_id.to_s if key_id
41
+
42
+ metadata.merge(envelope)
43
+ end
44
+
45
+ def self.decrypt(payload_hash, key_provider_or_key, topic: nil)
46
+ return payload_hash unless payload_hash.is_a?(Hash) && payload_hash[:_encrypted]
47
+ return payload_hash unless key_provider_or_key
48
+
49
+ key = resolve_decryption_key(payload_hash, key_provider_or_key, topic)
50
+ return payload_hash unless key
51
+
52
+ cipher = OpenSSL::Cipher.new(ALGORITHM)
53
+ cipher.decrypt
54
+ cipher.key = key
55
+ cipher.iv = Base64.strict_decode64(payload_hash[:_iv])
56
+ cipher.auth_tag = Base64.strict_decode64(payload_hash[:_auth_tag])
57
+
58
+ encrypted_bytes = Base64.strict_decode64(payload_hash[:_data])
59
+ decrypted_json = cipher.update(encrypted_bytes) + cipher.final
60
+
61
+ decrypted_data = JSON.parse(decrypted_json, symbolize_names: true)
62
+
63
+ metadata = payload_hash.select { |k, _| k.to_s.start_with?("_") }
64
+ clean_envelope_metadata!(metadata)
65
+
66
+ decrypted_data.merge(metadata)
67
+ rescue => e
68
+ raise DecryptionError, "Failed to decrypt payload. Error: #{e.message}. Offending payload: #{payload_hash.inspect}"
69
+ end
70
+
71
+ private
72
+
73
+ def self.resolve_encryption_key(provider_or_key, topic)
74
+ if provider_or_key.respond_to?(:key_for)
75
+ [provider_or_key.key_for(topic), provider_or_key.active_key_id_for(topic)]
76
+ else
77
+ [provider_or_key, nil]
78
+ end
79
+ end
80
+
81
+ def self.resolve_decryption_key(payload, provider_or_key, topic)
82
+ if provider_or_key.respond_to?(:key_for_id)
83
+ key_id = payload[:_key_id]
84
+ key_id ? provider_or_key.key_for_id(key_id) : provider_or_key.key_for(topic)
85
+ else
86
+ provider_or_key
87
+ end
88
+ end
89
+
90
+ def self.clean_envelope_metadata!(metadata)
91
+ metadata.delete(:_encrypted)
92
+ metadata.delete(:_iv)
93
+ metadata.delete(:_auth_tag)
94
+ metadata.delete(:_data)
95
+ metadata.delete(:_key_id)
96
+ end
97
+ end
98
+ end
@@ -0,0 +1,78 @@
1
+ # frozen_string_literal: true
2
+
3
+ module SharedBroker
4
+ module KeyProvider
5
+ class KeyNotFoundError < StandardError; end
6
+
7
+ # Static key provider for backward compatibility
8
+ class Static
9
+ def initialize(key)
10
+ raise ArgumentError, "Expected key to be a 32-byte String, got #{key.inspect}" unless key.is_a?(String)
11
+ @key = key
12
+ end
13
+
14
+ def key_for(_topic)
15
+ @key
16
+ end
17
+
18
+ def key_for_id(_key_id)
19
+ @key
20
+ end
21
+
22
+ def active_key_id_for(_topic)
23
+ nil
24
+ end
25
+ end
26
+
27
+ # Flexible registry for multiple keys and topic-based routing patterns
28
+ class Registry
29
+ def initialize(keys: {}, active_keys: {})
30
+ validate_inputs!(keys, active_keys)
31
+ @keys = keys.transform_keys(&:to_s)
32
+ @active_keys = active_keys.transform_keys(&:to_s)
33
+ end
34
+
35
+ def key_for(topic)
36
+ key_id = active_key_id_for(topic)
37
+ key_for_id(key_id)
38
+ end
39
+
40
+ def key_for_id(key_id)
41
+ return nil if key_id.nil?
42
+
43
+ key = @keys[key_id.to_s]
44
+ return key if key
45
+
46
+ raise KeyNotFoundError, "Key ID #{key_id.inspect} not found in registered keys. Available keys: #{@keys.keys.inspect}"
47
+ end
48
+
49
+ def active_key_id_for(topic)
50
+ topic_str = topic.to_s
51
+ return @active_keys[topic_str] if @active_keys.key?(topic_str)
52
+
53
+ pattern = find_matching_pattern(topic_str)
54
+ return @active_keys[pattern] if pattern
55
+
56
+ fallback_key_id
57
+ end
58
+
59
+ private
60
+
61
+ def validate_inputs!(keys, active_keys)
62
+ unless keys.is_a?(Hash) && active_keys.is_a?(Hash)
63
+ raise ArgumentError, "Expected keys and active_keys to be Hashes, got keys: #{keys.inspect}, active_keys: #{active_keys.inspect}"
64
+ end
65
+ end
66
+
67
+ def find_matching_pattern(topic_str)
68
+ @active_keys.keys.find do |pattern|
69
+ pattern != "*" && File.fnmatch?(pattern, topic_str)
70
+ end
71
+ end
72
+
73
+ def fallback_key_id
74
+ @active_keys["*"] || raise(KeyNotFoundError, "No active key configuration found for fallback '*' pattern")
75
+ end
76
+ end
77
+ end
78
+ end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module SharedBroker
4
- VERSION = "1.5.0"
4
+ VERSION = "1.6.0"
5
5
  end
data/lib/shared_broker.rb CHANGED
@@ -8,6 +8,7 @@ require_relative "shared_broker/schema_registry/providers/local"
8
8
  require_relative "shared_broker/schema_registry/providers/http"
9
9
  require_relative "shared_broker/validation"
10
10
  require_relative "shared_broker/cipher"
11
+ require_relative "shared_broker/key_provider"
11
12
  require_relative "shared_broker/concurrency/semaphore"
12
13
  require_relative "shared_broker/concurrency/limiter"
13
14
  require_relative "shared_broker/middleware_pipeline"
@@ -20,27 +21,29 @@ require_relative "shared_broker/adapters/redis"
20
21
 
21
22
  module SharedBroker
22
23
  class << self
23
- attr_accessor :encryption_key
24
+ attr_accessor :encryption_key, :key_provider
24
25
  end
25
26
 
26
27
  # Default key for development/test if not set
27
28
  @encryption_key = ENV.fetch("SHARED_BROKER_ENCRYPTION_KEY") { "a" * 32 }
29
+ @key_provider = nil
28
30
 
29
31
  class Client
30
32
  attr_reader :circuit_breaker, :middleware_pipeline, :adapters, :routing
31
33
 
32
- def initialize(adapter: nil, adapters: nil, routing: nil, circuit_breaker: nil, middlewares: nil)
34
+ def initialize(adapter: nil, adapters: nil, routing: nil, circuit_breaker: nil, middlewares: nil, key_provider: nil)
33
35
  setup_adapters(adapter: adapter, adapters: adapters, routing: routing)
34
36
  @circuit_breaker = circuit_breaker || CircuitBreaker.new
35
37
  resolved_middlewares = middlewares || [SharedBroker::Middlewares::OpenTelemetryPropagation.new]
36
38
  @middleware_pipeline = MiddlewarePipeline.new(resolved_middlewares)
39
+ @key_provider = key_provider
37
40
  end
38
41
 
39
42
  def publish(topic, message, correlation_id: nil)
40
43
  metadata = { correlation_id: correlation_id, operation: :publish }
41
44
  @middleware_pipeline.execute(topic, message, metadata) do
42
45
  SharedBroker::Validation.validate!(topic, message)
43
- encrypted_msg = SharedBroker::Cipher.encrypt(message, SharedBroker.encryption_key)
46
+ encrypted_msg = SharedBroker::Cipher.encrypt(message, active_key_provider, topic: topic)
44
47
 
45
48
  @circuit_breaker.run do
46
49
  resolve_adapter(topic).publish(topic, encrypted_msg, correlation_id: correlation_id)
@@ -57,7 +60,7 @@ module SharedBroker
57
60
 
58
61
  resolve_adapter(topic).subscribe(topic, queue_name, max_retries: max_retries, backoff_base: backoff_base) do |raw_message|
59
62
  limiter.run do
60
- decrypted_msg = SharedBroker::Cipher.decrypt(raw_message, SharedBroker.encryption_key)
63
+ decrypted_msg = SharedBroker::Cipher.decrypt(raw_message, active_key_provider, topic: topic)
61
64
  SharedBroker::Validation.validate!(topic, decrypted_msg)
62
65
 
63
66
  metadata = { correlation_id: decrypted_msg[:_correlation_id], operation: :subscribe, queue_name: queue_name }
@@ -70,6 +73,10 @@ module SharedBroker
70
73
 
71
74
  private
72
75
 
76
+ def active_key_provider
77
+ @key_provider || SharedBroker.key_provider || SharedBroker::KeyProvider::Static.new(SharedBroker.encryption_key)
78
+ end
79
+
73
80
  def setup_adapters(adapter: nil, adapters: nil, routing: nil)
74
81
  if adapter || (adapters.nil? && routing.nil?)
75
82
  validate_single_adapter!(adapter)
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: shared_broker
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.5.0
4
+ version: 1.6.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Wesley Lima
8
8
  autorequire:
9
9
  bindir: exe
10
10
  cert_chain: []
11
- date: 2026-06-10 00:00:00.000000000 Z
11
+ date: 2026-06-16 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: bunny
@@ -157,6 +157,7 @@ files:
157
157
  - lib/shared_broker/circuit_breaker.rb
158
158
  - lib/shared_broker/concurrency/limiter.rb
159
159
  - lib/shared_broker/concurrency/semaphore.rb
160
+ - lib/shared_broker/key_provider.rb
160
161
  - lib/shared_broker/middleware_pipeline.rb
161
162
  - lib/shared_broker/middlewares/open_telemetry_propagation.rb
162
163
  - lib/shared_broker/schema_registry.rb