|
1 | 1 | # Dice bot for Discord
|
2 | 2 | # Author: Humblemonk
|
3 |
| -# Version: 8.11.1 |
| 3 | +# Version: 9.0.0 |
4 | 4 | # Copyright (c) 2017. All rights reserved.
|
5 | 5 | # !/usr/bin/ruby
|
6 | 6 | # If you wish to run a single instance of this bot, please follow the "Manual Install" section of the readme!
|
|
23 | 23 |
|
24 | 24 | # open connection to sqlite db and set timeout to 10s if the database is busy
|
25 | 25 | if @launch_option == 'lite'
|
26 |
| - # do nothing |
| 26 | + puts 'Dice Maiden lite mode detected!' |
27 | 27 | else
|
28 | 28 | require 'sqlite3'
|
29 | 29 | $db = SQLite3::Database.new 'main.db'
|
|
51 | 51 | inc_cmd = lambda do |event|
|
52 | 52 | # Locking the thread to prevent messages going to the wrong server
|
53 | 53 | mutex.lock
|
| 54 | + response_array = [] |
54 | 55 | begin
|
55 |
| - @event_roll = event.options.values.join('') |
56 |
| - |
57 |
| - @do_tally_shuffle = false |
58 |
| - check_comment |
59 |
| - @roll_request = @event_roll.dup |
60 |
| - |
61 |
| - @input = alias_input_pass(@event_roll) # Do alias pass as soon as we get the message |
62 |
| - @simple_output = false |
63 |
| - @wng = false |
64 |
| - @dh = false |
65 |
| - @godbound = false |
66 |
| - @ed = false |
67 |
| - @hsn = false |
68 |
| - @hsk = false |
69 |
| - @hsh = false |
70 |
| - @no_result = false |
71 |
| - |
72 |
| - @private_roll = false |
73 |
| - @reroll_check = 0 |
74 |
| - @reroll_indefinite_check = 0 |
75 |
| - @reroll_count = 0 |
76 |
| - @botch = 0 |
77 |
| - |
78 |
| - check_roll_modes |
79 |
| - next if @ed && !replace_earthdawn(event) |
80 |
| - |
81 |
| - @roll_set = nil |
82 |
| - next unless roll_sets_valid(event) |
83 |
| - |
84 |
| - # check for single dice rolls |
85 |
| - @input.gsub!(%r{(?<!\d)(^|[+*/-]\s?)d(\d+)}, '\11d\2') if @input.match?(%r{(?<!\d)(^|[+*/-]\s?)d(\d+)}) |
86 |
| - |
87 |
| - @roll = @input |
88 |
| - @check = @prefix + @roll |
89 |
| - @test_status = '' |
90 |
| - # check user |
91 |
| - check_user_or_nick(event) |
92 |
| - # check for empty roll |
93 |
| - if @event_roll.empty? |
94 |
| - event.respond(content: "#{@user} roll is empty! Please type a complete dice roll message") |
95 |
| - next |
96 |
| - end |
97 |
| - # check for modifiers that should apply to everything |
98 |
| - check_universal_modifiers |
99 |
| - |
100 |
| - # Check for dn |
101 |
| - dnum = @input.scan(/dn\s?(\d+)/).first.join.to_i if @input.match?(/^(1dn)\d+/i) |
102 |
| - |
103 |
| - # Check for correct input |
104 |
| - if @roll.match?(/\dd\d/i) |
105 |
| - event.channel.start_typing |
106 |
| - next if check_roll(event) == true |
107 |
| - |
108 |
| - # Check for wrath roll |
109 |
| - check_wrath |
110 |
| - # Grab dice roll, create roll, grab results |
111 |
| - if @roll_set.nil? |
112 |
| - next if do_roll(event) == true |
113 |
| - else |
114 |
| - @roll_set_results = '' |
115 |
| - @error_check_roll_set = '' |
116 |
| - roll_count = 0 |
117 |
| - @roll_set_total = 0 |
118 |
| - error_encountered = false |
119 |
| - while roll_count < @roll_set.to_i |
120 |
| - if do_roll(event) == true |
121 |
| - error_encountered = true |
122 |
| - break |
| 56 | + inc_event_roll = event.options.values.join('') |
| 57 | + rolls_array = inc_event_roll.split(%r{\s*/\s*}).take(4) |
| 58 | + rolls_array.each do |event_roll| |
| 59 | + @do_tally_shuffle = false |
| 60 | + check_comment(event_roll) |
| 61 | + @roll_request = event_roll.dup |
| 62 | + |
| 63 | + @input = alias_input_pass(event_roll) # Do alias pass as soon as we get the message |
| 64 | + @simple_output = false |
| 65 | + @wng = false |
| 66 | + @dh = false |
| 67 | + @godbound = false |
| 68 | + @ed = false |
| 69 | + @hsn = false |
| 70 | + @hsk = false |
| 71 | + @hsh = false |
| 72 | + @no_result = false |
| 73 | + |
| 74 | + @private_roll = false |
| 75 | + @reroll_check = 0 |
| 76 | + @reroll_indefinite_check = 0 |
| 77 | + @reroll_count = 0 |
| 78 | + @botch = 0 |
| 79 | + |
| 80 | + check_roll_modes |
| 81 | + next if @ed && !replace_earthdawn(event) |
| 82 | + |
| 83 | + @roll_set = nil |
| 84 | + next unless roll_sets_valid(event) |
| 85 | + |
| 86 | + # check for single dice rolls |
| 87 | + @input.gsub!(%r{(?<!\d)(^|[+*/-]\s?)d(\d+)}, '\11d\2') if @input.match?(%r{(?<!\d)(^|[+*/-]\s?)d(\d+)}) |
| 88 | + |
| 89 | + @roll = @input |
| 90 | + @check = @prefix + @roll |
| 91 | + @test_status = '' |
| 92 | + # check user |
| 93 | + check_user_or_nick(event) |
| 94 | + # check for empty roll |
| 95 | + if event_roll.empty? |
| 96 | + event.respond(content: "#{@user} roll is empty! Please type a complete dice roll message") |
| 97 | + next |
| 98 | + end |
| 99 | + # check for modifiers that should apply to everything |
| 100 | + check_universal_modifiers |
| 101 | + |
| 102 | + # Check for dn |
| 103 | + @dnum = @input.scan(/dn\s?(\d+)/).first.join.to_i if @input.match?(/^(1dn)\d+/i) |
| 104 | + |
| 105 | + # Check for correct input |
| 106 | + if @roll.match?(/\dd\d/i) |
| 107 | + event.channel.start_typing |
| 108 | + next if check_roll(event) == true |
| 109 | + |
| 110 | + # Check for wrath roll |
| 111 | + check_wrath |
| 112 | + # Grab dice roll, create roll, grab results |
| 113 | + if @roll_set.nil? |
| 114 | + next if do_roll(event) == true |
| 115 | + else |
| 116 | + @roll_set_results = '' |
| 117 | + @error_check_roll_set = '' |
| 118 | + roll_count = 0 |
| 119 | + @roll_set_total = 0 |
| 120 | + error_encountered = false |
| 121 | + while roll_count < @roll_set.to_i |
| 122 | + if do_roll(event) == true |
| 123 | + error_encountered = true |
| 124 | + break |
| 125 | + end |
| 126 | + @tally = alias_output_pass(@tally) |
| 127 | + if @simple_output == true |
| 128 | + @roll_set_results << "#{@dice_result}\n" |
| 129 | + else |
| 130 | + @error_check_roll_set << "#{@dice_result}\n" |
| 131 | + @roll_set_results << "`#{@tally}` #{@dice_result}\n" |
| 132 | + end |
| 133 | + roll_count += 1 |
123 | 134 | end
|
124 |
| - @tally = alias_output_pass(@tally) |
125 |
| - if @simple_output == true |
126 |
| - @roll_set_results << "#{@dice_result}\n" |
| 135 | + next if error_encountered |
| 136 | + |
| 137 | + log_roll(event) if @launch_option == 'debug' |
| 138 | + if @comment.to_s.empty? || @comment.to_s.nil? |
| 139 | + event.respond(content: "#{@user} Request: `[#{@roll_request.strip}]` Rolls:\n#{@roll_set_results}Results Total: `#{@roll_set_total}`") |
127 | 140 | else
|
128 |
| - @error_check_roll_set << "#{@dice_result}\n" |
129 |
| - @roll_set_results << "`#{@tally}` #{@dice_result}\n" |
| 141 | + event.respond(content: "#{@user} Rolls:\n#{@roll_set_results}Results Total: `#{@roll_set_total}`\nReason: `#{@comment}`") |
130 | 142 | end
|
131 |
| - roll_count += 1 |
| 143 | + next |
132 | 144 | end
|
133 |
| - next if error_encountered |
134 | 145 |
|
135 |
| - log_roll(event) if @launch_option == 'debug' |
136 |
| - if @comment.to_s.empty? || @comment.to_s.nil? |
137 |
| - event.respond(content: "#{@user} Request: `[#{@roll_request.strip}]` Rolls:\n#{@roll_set_results}Results Total: `#{@roll_set_total}`") |
138 |
| - else |
139 |
| - event.respond(content: "#{@user} Rolls:\n#{@roll_set_results}Results Total: `#{@roll_set_total}`\nReason: `#{@comment}`") |
140 |
| - end |
141 |
| - next |
142 |
| - end |
| 146 | + # Output aliasing |
| 147 | + @tally = alias_output_pass(@tally) |
143 | 148 |
|
144 |
| - # Output aliasing |
145 |
| - @tally = alias_output_pass(@tally) |
| 149 | + # Does calculation for Hero System stuff, if necessary |
| 150 | + hero_system_math if @hsn || @hsk |
146 | 151 |
|
147 |
| - # Does calculation for Hero System stuff, if necessary |
148 |
| - hero_system_math if @hsn || @hsk |
| 152 | + # Grab event user name, server name and timestamp for roll and log it |
| 153 | + log_roll(event) if @launch_option == 'debug' |
149 | 154 |
|
150 |
| - # Grab event user name, server name and timestamp for roll and log it |
151 |
| - log_roll(event) if @launch_option == 'debug' |
| 155 | + @has_comment = !@comment.to_s.empty? && !@comment.to_s.nil? |
152 | 156 |
|
153 |
| - # Print dice result to Discord channel |
154 |
| - @has_comment = !@comment.to_s.empty? && !@comment.to_s.nil? |
155 |
| - if check_wrath == true |
156 |
| - respond_wrath(event, dnum) |
157 |
| - elsif @private_roll |
158 |
| - event.respond(content: build_response, ephemeral: true) |
159 |
| - else |
160 |
| - event.respond(content: build_response) |
161 |
| - check_fury(event) |
| 157 | + response_array.push(build_response) |
162 | 158 | end
|
163 |
| - end |
164 |
| - next if check_donate(event) == true |
165 |
| - next if check_help(event) == true |
166 |
| - next if check_bot_info(event) == true |
167 |
| - next if check_purge(event) == false |
168 |
| - rescue StandardError => e ## The worst that should happen is that we catch the error and return its message. |
169 |
| - e.message = 'NIL MESSAGE!' if e.message.nil? |
170 |
| - # Simplify roll and send it again if we error out due to character limit |
171 |
| - if (e.message.include? 'Message over the character limit') || (e.message.include? 'Invalid Form Body') |
172 |
| - if @roll_set.nil? |
173 |
| - event.respond(content: "#{@user} Roll #{@dice_result} Reason: `Simplified roll due to character limit`") |
| 159 | + next if check_donate(event) == true |
| 160 | + next if check_help(event) == true |
| 161 | + next if check_bot_info(event) == true |
| 162 | + next if check_purge(event) == false |
| 163 | + rescue StandardError => e ## The worst that should happen is that we catch the error and return its message. |
| 164 | + e.message = 'NIL MESSAGE!' if e.message.nil? |
| 165 | + # Simplify roll and send it again if we error out due to character limit |
| 166 | + if (e.message.include? 'Message over the character limit') || (e.message.include? 'Invalid Form Body') |
| 167 | + if @roll_set.nil? |
| 168 | + event.respond(content: "#{@user} Roll #{@dice_result} Reason: `Simplified roll due to character limit`") |
| 169 | + else |
| 170 | + event.respond(content: "#{@user} Rolls:\n#{@error_check_roll_set}Reason: `Simplified roll due to character limit`") |
| 171 | + end |
| 172 | + elsif (e.message.include? "undefined method `join' for nil:NilClass") || (e.message.include? "The bot doesn't have the required permission to do this!") || (e.message.include? '500: Internal Server Error') || (e.message.include? '500 Internal Server Error') |
| 173 | + time = Time.now.getutc |
| 174 | + File.open('dice_rolls.log', 'a') { |f| f.puts "#{time} ERROR: #{e.message}" } |
174 | 175 | else
|
175 |
| - event.respond(content: "#{@user} Rolls:\n#{@error_check_roll_set}Reason: `Simplified roll due to character limit`") |
| 176 | + event.respond(content: ('Unexpected exception thrown! (' + e.message + ")\n\nPlease drop us a message in the #support channel on the dice maiden server, or create an issue on Github.")) |
176 | 177 | end
|
177 |
| - elsif (e.message.include? "undefined method `join' for nil:NilClass") || (e.message.include? "The bot doesn't have the required permission to do this!") || (e.message.include? '500: Internal Server Error') || (e.message.include? '500 Internal Server Error') |
178 |
| - time = Time.now.getutc |
179 |
| - File.open('dice_rolls.log', 'a') { |f| f.puts "#{time} ERROR: #{e.message}" } |
180 |
| - else |
181 |
| - event.respond(content: ('Unexpected exception thrown! (' + e.message + ")\n\nPlease drop us a message in the #support channel on the dice maiden server, or create an issue on Github.")) |
182 | 178 | end
|
183 | 179 | end
|
| 180 | + # Print dice results to Discord channel |
| 181 | + # reduce noisy errors by checking if response array is empty due to responding earlier |
| 182 | + if response_array.empty? |
| 183 | + # do nothing |
| 184 | + elsif check_wrath == true |
| 185 | + respond_wrath(event, @dnum) |
| 186 | + elsif @private_roll |
| 187 | + event.respond(content: response_array.join("\n").to_s, ephemeral: true) |
| 188 | + else |
| 189 | + event.respond(content: response_array.join("\n").to_s) |
| 190 | + check_fury(event) |
| 191 | + end |
184 | 192 | mutex.unlock
|
185 | 193 | end
|
186 | 194 |
|
|
0 commit comments