forest_admin_rpc_agent 1.3.0 → 1.4.1

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: 1864c16d481ba458e77c25c92087142682e01fad811480131e45998ed81debe6
4
- data.tar.gz: 3d0e960bed5b8c798f28fe1b65de12654de3989d88cea6fbee56461a8043fc5b
3
+ metadata.gz: c41c96adbed57c2146f5e9c6ef71faecd39f1a7544a03637a1ee693d81429f9a
4
+ data.tar.gz: cf07504184989d2962c3df1101ee2dfc180d9e1d75ad538894019d6f4b8528c1
5
5
  SHA512:
6
- metadata.gz: 36b11ef9fd6cda133cfcb1c85217930fe3c578352b838c100aeb9d26893a89d9a019a22a8ccb95468b23090f91ccb0f387aec25c8d54691d16d6cc29d902b5c9
7
- data.tar.gz: fc149936c9c35a65d5fb6141efc16de5fae2d240d20e57da5236f9e1bc22b6aeed4bda887dd31716e1af9e918c6cd0d1d71c72387f5b1639bb00c4043591071c
6
+ metadata.gz: 7ec6bb5197fc7720ecf593e207141343ae29fb873e01ea2e68c00aa744fb1ba86aa8d8906e647ab27799f336d90cba01f5a45b3939a9abf0c5930340ece1b531
7
+ data.tar.gz: 037b674c70759f540e6cfea09fb86ffdfbce66faa060bd827857e49b2c113713dc9534963b64f1f4e580eb741d6637810e309d4641109655b1982c33852cfeb7
@@ -4,6 +4,7 @@ module ForestAdminRpcAgent
4
4
  ALLOWED_TIME_DIFF = 300
5
5
  SIGNATURE_REUSE_WINDOW = 5
6
6
  @@used_signatures = {}
7
+ @@signatures_mutex = Mutex.new
7
8
 
8
9
  def initialize(app)
9
10
  @app = app
@@ -32,33 +33,45 @@ module ForestAdminRpcAgent
32
33
  return false unless Rack::Utils.secure_compare(signature, expected_signature)
33
34
 
34
35
  # check if this signature has already been used (replay attack)
35
- if @@used_signatures.key?(signature)
36
- last_used = @@used_signatures[signature]
37
- return false if Time.now.utc.to_i - last_used > SIGNATURE_REUSE_WINDOW
38
- end
39
- @@used_signatures[signature] = Time.now.utc.to_i
36
+ # Reject if signature was used recently (within SIGNATURE_REUSE_WINDOW seconds)
37
+ # Use mutex to prevent race conditions in multi-threaded environments
38
+ now = current_time_in_seconds
39
+
40
+ @@signatures_mutex.synchronize do
41
+ if @@used_signatures.key?(signature)
42
+ last_used = @@used_signatures[signature]
43
+ time_since_last_use = now - last_used
44
+ return false if time_since_last_use <= SIGNATURE_REUSE_WINDOW
45
+ end
46
+ @@used_signatures[signature] = now
40
47
 
41
- cleanup_old_signatures
48
+ cleanup_old_signatures
49
+ end
42
50
 
43
51
  true
44
52
  end
45
53
 
46
54
  def valid_timestamp?(timestamp)
47
55
  time = begin
48
- Time.iso8601(timestamp)
49
- rescue StandardError
56
+ Time.iso8601(timestamp).utc
57
+ rescue ArgumentError
50
58
  nil
51
59
  end
52
60
  return false if time.nil?
53
61
 
54
- (Time.now.utc.to_i - time.to_i).abs <= ALLOWED_TIME_DIFF
62
+ (current_time_in_seconds - time.to_i).abs <= ALLOWED_TIME_DIFF
55
63
  end
56
64
 
57
65
  def cleanup_old_signatures
58
- now = Time.now.utc.to_i
66
+ # Should be called within mutex synchronize block
67
+ now = current_time_in_seconds
59
68
  @@used_signatures.delete_if { |_signature, last_used| now - last_used > ALLOWED_TIME_DIFF }
60
69
  end
61
70
 
71
+ def current_time_in_seconds
72
+ defined?(Time.current) ? Time.current.to_i : Time.now.utc.to_i
73
+ end
74
+
62
75
  def auth_secret
63
76
  ForestAdminRpcAgent.config.auth_secret
64
77
  end
@@ -0,0 +1,22 @@
1
+ require 'jsonapi-serializers'
2
+
3
+ module ForestAdminRpcAgent
4
+ module Routes
5
+ class NativeQuery < BaseRoute
6
+ def initialize
7
+ super('rpc/native-query', 'post', 'rpc_native_query')
8
+ end
9
+
10
+ def handle_request(args)
11
+ return '{}' unless args[:params]['connection_name'] && args[:params]['query']
12
+
13
+ connection_name = args[:params]['connection_name']
14
+ query = args[:params]['query']
15
+ binds = args[:params]['binds'] || []
16
+ datasource = ForestAdminRpcAgent::Facades::Container.datasource
17
+
18
+ datasource.execute_native_query(connection_name, query, binds).to_json
19
+ end
20
+ end
21
+ end
22
+ end
@@ -12,10 +12,19 @@ module ForestAdminRpcAgent
12
12
  def handle_request(_params)
13
13
  agent = ForestAdminRpcAgent::Agent.instance
14
14
  schema = agent.customizer.schema
15
- schema[:collections] = agent.customizer.datasource(ForestAdminRpcAgent::Facades::Container.logger)
16
- .collections
17
- .map { |_name, collection| collection.schema.merge({ name: collection.name }) }
18
- .sort_by { |collection| collection[:name] }
15
+ datasource = agent.customizer.datasource(ForestAdminRpcAgent::Facades::Container.logger)
16
+
17
+ schema[:collections] = datasource.collections
18
+ .map { |_name, collection| collection.schema.merge({ name: collection.name }) }
19
+ .sort_by { |collection| collection[:name] }
20
+
21
+ connections = []
22
+ agent.customizer.datasources.each do |root_datasource|
23
+ connections = connections.union(
24
+ root_datasource.live_query_connections.keys.map { |connection_name| { name: connection_name } }
25
+ )
26
+ end
27
+ schema[:nativeQueryConnections] = connections
19
28
 
20
29
  schema.to_json
21
30
  end
@@ -1,3 +1,3 @@
1
1
  module ForestAdminRpcAgent
2
- VERSION = "1.3.0"
2
+ VERSION = "1.4.1"
3
3
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: forest_admin_rpc_agent
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.3.0
4
+ version: 1.4.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Matthieu
@@ -103,6 +103,7 @@ files:
103
103
  - lib/forest_admin_rpc_agent/routes/delete.rb
104
104
  - lib/forest_admin_rpc_agent/routes/health_route.rb
105
105
  - lib/forest_admin_rpc_agent/routes/list.rb
106
+ - lib/forest_admin_rpc_agent/routes/native_query.rb
106
107
  - lib/forest_admin_rpc_agent/routes/schema.rb
107
108
  - lib/forest_admin_rpc_agent/routes/sse.rb
108
109
  - lib/forest_admin_rpc_agent/routes/update.rb