simplex-chat 0.6.0 → 0.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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 0b49dea10199c71c10df356701b53087eda1e8aeb5c31348e39a62032280d96b
4
- data.tar.gz: e97b4de663eb4abe3aac556b6f8a1b2cf1dcce1d37b2ebfdfc645a7b8e9c05a4
3
+ metadata.gz: 662e4bdcc4df337d7382a5b1c13dc3ac7f9f89db0ea69d5b97c778f20e0d841c
4
+ data.tar.gz: c9fba2eb82c77f0e35ff434864b4a8443edd41662697f4de46f1c10dd412d059
5
5
  SHA512:
6
- metadata.gz: c2c78755a44732f0b2ab43a080981d4f332c414e2080ff8e4d8ddf76a6af8ed53a0e6c5699561df33b0d8e9d713c92774c1dbff75cc60809adb4ff463f10af8f
7
- data.tar.gz: 70eb7367381d1442ed72ff4ae92ed7d876d99e1d3d506bd4da9eafa833f7e1e32c814bec37e052ca9c0b932f726bba59668459de646fac20b004fff14f0ced15
6
+ metadata.gz: c2f3f4895711d3e1e26b1b1cb843aee645fac8b52d4b6f57621f0ac1ad6966363a9130eea9b48af896d2640547792e9577ae25e47a096d48bbf293d6b634c00f
7
+ data.tar.gz: f5a533224f6058317f47fef4c65874320823dd55517282c84ce30d31d438d365f00e968a39bf688c009a58be68b47fe130e1c76bbba2def81cf11cb46b15b739
@@ -0,0 +1,163 @@
1
+ module SimpleXChat
2
+ class BasicCommand
3
+ attr_reader :name, :num_args, :desc, :min_role
4
+
5
+ # TODO: Allow optional arguments
6
+ def initialize(name, desc="", num_args: 0, min_role: GroupMemberRole::MEMBER,
7
+ per_sender_cooldown_secs: nil, per_issuer_cooldown_secs: nil)
8
+ @name = name
9
+ @num_args = num_args
10
+ @desc = desc
11
+ @min_role = min_role
12
+ @per_sender_cooldown_secs = per_sender_cooldown_secs
13
+ @per_issuer_cooldown_secs = per_issuer_cooldown_secs
14
+ @last_runs = {
15
+ :per_sender => {},
16
+ :per_issuer => {}
17
+ }
18
+ @last_runs_lock = Mutex.new
19
+ end
20
+
21
+ def validate_and_execute(client, chat_msg, args)
22
+ return if not validate(client, chat_msg, args)
23
+ execute client, chat_msg, args
24
+ end
25
+
26
+ private
27
+
28
+ def validate(client, chat_msg, args)
29
+ msg_text = chat_msg[:msg_text]
30
+ chat_type = chat_msg[:chat_type]
31
+ issuer = chat_msg[:contact]
32
+ issuer_role = chat_msg[:contact_role]
33
+ sender = chat_msg[:sender]
34
+
35
+ # Verify that user has permissions to run the command
36
+ role_hierarchy = {
37
+ GroupMemberRole::MEMBER => 0,
38
+ GroupMemberRole::ADMIN => 1,
39
+ GroupMemberRole::OWNER => 2
40
+ }
41
+ perms = role_hierarchy[issuer_role]
42
+ if issuer_role != nil and perms == nil || perms < role_hierarchy[@min_role]
43
+ client.api_send_text_message chat_type, sender, "@#{issuer}: You do not have permission to run this command (required: #{@min_role})"
44
+ return false
45
+ end
46
+
47
+ # Verify arguments
48
+ if args.length != @num_args
49
+ client.api_send_text_message chat_type, sender, "@#{issuer}: Incorrect number of arguments (required: #{@num_args})"
50
+ return false
51
+ end
52
+
53
+ # Verify per sender cooldown
54
+ # NOTE: This should be the last verification, because
55
+ # it will update the last-validated-runs object
56
+ is_on_cooldown = true
57
+ remaining_cooldown = 0.0
58
+ chat = "#{chat_type}#{sender}"
59
+ chat_and_issuer = "#{chat_type}#{sender}[#{issuer}]"
60
+ @last_runs_lock.synchronize {
61
+ sender_last_run = @last_runs[:per_sender][chat]
62
+ issuer_last_run = @last_runs[:per_issuer][chat_and_issuer]
63
+ now = Time.now
64
+ if sender_last_run != nil && @per_sender_cooldown_secs != nil
65
+ time_diff = now - sender_last_run
66
+ if time_diff < @per_sender_cooldown_secs
67
+ remaining_cooldown = @per_sender_cooldown_secs - time_diff
68
+ end
69
+ end
70
+
71
+ if issuer_last_run != nil && @per_issuer_cooldown_secs != nil
72
+ time_diff = now - issuer_last_run
73
+ if time_diff < @per_issuer_cooldown_secs
74
+ cooldown = @per_issuer_cooldown_secs - time_diff
75
+ remaining_cooldown = [cooldown, remaining_cooldown].max
76
+ end
77
+ end
78
+
79
+ break if remaining_cooldown > 0.0
80
+
81
+ @last_runs[:per_sender][chat] = now
82
+ @last_runs[:per_issuer][chat_and_issuer] = now
83
+ is_on_cooldown = false
84
+ }
85
+
86
+ if is_on_cooldown
87
+ client.api_send_text_message chat_type, sender, "@#{issuer}: On cooldown, try again in #{remaining_cooldown.round(1)} seconds"
88
+ return false
89
+ end
90
+
91
+
92
+ return true
93
+ end
94
+
95
+ def execute(client, chat_msg, args)
96
+ raise NoMethodError.new(
97
+ "[!] Default BasicCommand::execute called, nothing will be done\n" \
98
+ " Extend this class to implement custom execution behavior"
99
+ )
100
+ end
101
+ end
102
+
103
+ class BasicCommandRunner
104
+ def initialize(client, commands, prefix)
105
+ @client = client
106
+ @commands = commands.map { |cmd|
107
+ { "#{prefix}#{cmd.name}" => cmd }
108
+ }.reduce({}, &:merge)
109
+ @prefix = prefix
110
+ @logger = Logging.logger
111
+ end
112
+
113
+ def listen(max_backlog_secs: 5.0)
114
+ loop do
115
+ begin
116
+ break if process_next_event(max_backlog_secs) == :stop
117
+ rescue SimpleXChat::GenericError => e
118
+ @logger.error("[!] Caught error: #{e}")
119
+ rescue => e
120
+ raise e
121
+ end
122
+ end
123
+ end
124
+
125
+ private
126
+
127
+ def process_next_event(max_backlog_secs)
128
+ chat_msg = @client.next_chat_message(max_backlog_secs: max_backlog_secs)
129
+ if chat_msg == nil
130
+ @logger.warn("Message queue is closed")
131
+ return :stop
132
+ end
133
+ @logger.debug("Chat message: #{chat_msg}")
134
+
135
+ msg_text = chat_msg[:msg_text]
136
+ chat_type = chat_msg[:chat_type]
137
+ issuer = chat_msg[:contact]
138
+ issuer_role = chat_msg[:contact_role]
139
+ sender = chat_msg[:sender]
140
+
141
+ # Verify if this is a registered command
142
+ message_items = msg_text.split(" ")
143
+ first_word = message_items[0]
144
+
145
+ # React to all messages we will process
146
+ if first_word.start_with?(@prefix)
147
+ @client.api_reaction chat_msg[:chat_type], chat_msg[:sender_id], chat_msg[:msg_item_id], emoji: '🚀'
148
+ end
149
+
150
+ command = @commands[first_word]
151
+ if command == nil
152
+ @client.api_send_text_message chat_msg[:chat_type], chat_msg[:sender], "@#{issuer}: Unknown command"
153
+ return
154
+ end
155
+
156
+ args = message_items[1..]
157
+
158
+ # Run command
159
+ @logger.debug("Validating and executing command '#{command.name}' for: #{chat_type}#{sender} [#{issuer}]: #{msg_text}")
160
+ command.validate_and_execute @client, chat_msg, args
161
+ end
162
+ end
163
+ end
@@ -0,0 +1,18 @@
1
+ require 'logger'
2
+
3
+ module SimpleXChat
4
+ module Logging
5
+ def self.logger(dest: $stderr, log_level: Logger::INFO)
6
+ if @logger == nil
7
+ @logger = Logger.new(dest)
8
+ @logger.level = log_level
9
+ @logger.progname = 'simplex-chat'
10
+ @logger.formatter = -> (severity, datetime, progname, msg) {
11
+ "| [#{severity}] | #{datetime} | (#{progname}) :: #{msg}\n"
12
+ }
13
+ end
14
+
15
+ @logger
16
+ end
17
+ end
18
+ end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module SimpleXChat
4
- VERSION = "0.6.0"
4
+ VERSION = "0.7.0"
5
5
  end
data/lib/simplex-chat.rb CHANGED
@@ -4,6 +4,7 @@ require_relative 'simplex-chat/version'
4
4
  require_relative 'simplex-chat/errors'
5
5
  require_relative 'simplex-chat/patches'
6
6
  require_relative 'simplex-chat/types'
7
+ require_relative 'simplex-chat/logger'
7
8
 
8
9
  module SimpleXChat
9
10
  require 'net/http'
@@ -14,9 +15,7 @@ module SimpleXChat
14
15
  require 'time'
15
16
 
16
17
  class ClientAgent
17
- attr_accessor :on_message
18
-
19
- def initialize(client_uri, connect: true, log_level: Logger::INFO, timeout_ms: 10_000, interval_ms: 100)
18
+ def initialize(client_uri, connect: true, timeout_ms: 10_000, interval_ms: 100)
20
19
  @uri = client_uri
21
20
  @message_queue = Queue.new
22
21
  @chat_message_queue = Queue.new
@@ -30,12 +29,7 @@ module SimpleXChat
30
29
  @timeout_ms = timeout_ms
31
30
  @interval_ms = interval_ms
32
31
 
33
- @logger = Logger.new($stderr)
34
- @logger.level = log_level
35
- @logger.progname = 'simplex-chat'
36
- @logger.formatter = -> (severity, datetime, progname, msg) {
37
- "| [#{severity}] | #{datetime} | (#{progname}) :: #{msg}\n"
38
- }
32
+ @logger = Logging.logger
39
33
 
40
34
  self.connect if connect
41
35
 
@@ -110,7 +104,13 @@ module SimpleXChat
110
104
  # a chat message queue to insert one or
111
105
  # more messages at a time, but poll just
112
106
  # one at a time
113
- return @chat_message_queue.pop if not @chat_message_queue.empty?
107
+ # NOTE: We use non-blocking pop for thread safety
108
+ begin
109
+ chat_msg = @chat_message_queue.pop(true)
110
+ return chat_msg
111
+ rescue ThreadError
112
+ @logger.debug("Chat message queue is empty, waiting for new messages...")
113
+ end
114
114
 
115
115
  loop do
116
116
  msg = next_message
@@ -135,13 +135,20 @@ module SimpleXChat
135
135
  end
136
136
 
137
137
  @chat_message_queue.push chat_message
138
+ @logger.debug("Chat message pushed to queue: #{chat_message}")
138
139
  end
139
140
 
140
141
  # NOTE: Even after parsing the messages, the
141
142
  # chat message queue can be empty because
142
143
  # all the messages are too old, so we have
143
144
  # to check again
144
- return @chat_message_queue.pop if not @chat_message_queue.empty?
145
+ begin
146
+ chat_msg = @chat_message_queue.pop(true)
147
+ @logger.debug("Chat message popped from queue")
148
+ return chat_msg
149
+ rescue ThreadError
150
+ @logger.debug("Chat message queue is still empty, waiting for new messages...")
151
+ end
145
152
  end
146
153
 
147
154
  nil
metadata CHANGED
@@ -1,13 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: simplex-chat
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.6.0
4
+ version: 0.7.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - rdbo
8
8
  bindir: exe
9
9
  cert_chain: []
10
- date: 2025-03-05 00:00:00.000000000 Z
10
+ date: 2025-03-12 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: websocket
@@ -48,7 +48,9 @@ files:
48
48
  - README.md
49
49
  - Rakefile
50
50
  - lib/simplex-chat.rb
51
+ - lib/simplex-chat/cmd-runner.rb
51
52
  - lib/simplex-chat/errors.rb
53
+ - lib/simplex-chat/logger.rb
52
54
  - lib/simplex-chat/patches.rb
53
55
  - lib/simplex-chat/types.rb
54
56
  - lib/simplex-chat/version.rb