-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathMessage.lua
422 lines (342 loc) · 15.1 KB
/
Message.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
--========================================================--
-- Scorpio Message System --
-- --
-- Author : kurapica125@outlook.com --
-- Create Date : 2021/09/05 --
--========================================================--
--========================================================--
Scorpio "Scorpio.Message" ""
--========================================================--
SCORPIO_ADDON_PREFIX = "SCORPIO"
MESSAGE_OVERHEAD = 40
MAX_BYTE_SECOND = 2000
MAX_AVAILABLE = 8000
_WorldBeginTime = 0
_Available = 0
_LastUpdate = 0
_Recycle = Recycle()
_ProcessQueue = false
_AddonMessageQueue = Queue()
_ResultMessages = {}
_ModulePrefixHandler = {}
_NextMessageTasks = {}
------------------------------------------------------------
-- Helpers --
------------------------------------------------------------
export {
min = math.min, ceil = math.ceil, floor = math.floor,
random = math.random, format = string.format, char = string.char, byte = string.byte,
concat = table.concat, tinsert = table.insert,
yield = coroutine.yield, resume = coroutine.resume, running = coroutine.running
}
function updateAvailable()
local now = GetTime()
_Available = min(MAX_AVAILABLE, _Available + floor(MAX_BYTE_SECOND * (now - _LastUpdate)))
_LastUpdate = now
end
function toNumber(str)
local b1, b2 = byte(str, 1, 2)
return (b1 - 65) * 26 + b2 - 65
end
function toString(index)
return char(65 + floor(index / 26), 65 + index % 26)
end
__Async__()
function nextMessageCall(prefix, callback, timeout)
return callback(NextMessage(prefix, timeout))
end
function releaseNextMessage(task)
if task[2] then
local handlers = _NextMessageTasks[prefix]
if handlers then
local length = #handlers
for i = 1, length do
if handlers[i] == task then
handlers[i] = false
if length == 1 then _NextMessageTasks[prefix] = nil end
break
end
end
end
local ok, err = resume(task[2])
if not ok then geterrorhandler()(err) end
task[2] = nil
end
end
------------------------------------------------------------
-- Message Enumeration --
------------------------------------------------------------
if Scorpio.IsRetail then
__Sealed__() enum "ChatType" { "PARTY", "RAID", "INSTANCE_CHAT", "GUILD", "OFFICER", "WHISPER", "CHANNEL" }
else
__Sealed__() enum "ChatType" { "PARTY", "RAID", "INSTANCE_CHAT", "GUILD", "OFFICER", "WHISPER", "SAY", "YELL" }
end
------------------------------------------------------------
-- Message Event Handler --
------------------------------------------------------------
function OnEnable()
OnEnable = nil
C_ChatInfo.RegisterAddonMessagePrefix(SCORPIO_ADDON_PREFIX)
end
__SystemEvent__()
function PLAYER_ENTERING_WORLD()
_WorldBeginTime = GetTime()
_LastUpdate = GetTime()
_Available = 0
end
__SecureHook__"SendChatMessage"
function Hook_SendChatMessage(message, chatType, language, target)
_Available = _Available - (#tostring(message or "") + #tostring(target or "") + MESSAGE_OVERHEAD)
end
__SecureHook__(_G.C_ChatInfo, "SendAddonMessage")
function Hook_SendAddonMessage(prefix, message, chatType, target)
_Available = _Available - (#tostring(prefix or "") + #tostring(message or "") + #tostring(target or "") + MESSAGE_OVERHEAD)
end
__Async__()
function DistributeMessages(packet, channel, sender, target)
local data = Toolset.parsestring(DeflateDecode(Base64Decode(concat(packet))))
if data then
for _, item in ipairs(data) do
local prefix = item[1]
local message = item[2] and Toolset.parsestring(item[2])
if prefix and message then
local handlers = _ModulePrefixHandler[prefix]
if handlers then
for owner, handler in pairs(handlers) do
local ok, err = pcall(handler, message, channel, sender, target)
if not ok then geterrorhandler()(err) end
end
end
handlers = _NextMessageTasks[prefix]
_NextMessageTasks[prefix] = nil
if handlers then
Continue()
for _, thread in ipairs(handlers) do
if type(thread) == "table" then
-- With timeout, no recycle, just let it be collected
local temp = thread
thread = thread[2]
temp[2] = nil
end
if thread then
local ok, err = resume(thread, message, channel, sender, target)
if not ok then geterrorhandler()(err) end
end
end
_Recycle(wipe(handlers))
end
Continue()
end
end
end
end
__SystemEvent__()
function CHAT_MSG_ADDON(prefix, text, channel, sender, target, zoneChannelID, localID, name, instanceID)
if prefix ~= SCORPIO_ADDON_PREFIX then return end
local pid, count, id, msg = text:match("^(%w+):(%w+):(%w+):(.*)$")
pid = pid and tonumber(pid, 16)
count = count and toNumber(count)
id = id and toNumber(id)
if pid and count and id then
local now = GetTime()
local dist = _ResultMessages[pid]
if dist and ((now - dist.time) >= 10 or dist.count ~= count or (#dist + 1 ~= id)) then
dist = nil
_ResultMessages[pid]= nil
end
if not dist then
if id == 1 then
dist = { time = now, count = count, [1] = msg }
_ResultMessages[pid] = dist
else
return -- Abandon, no more check, no ack for now
end
else
dist[#dist + 1] = msg
end
if id == dist.count then
-- All received
_ResultMessages[pid]= nil
DistributeMessages(dist, channel, sender, target)
end
end
end
__Iterator__()
function iterdist(packets)
local yield = yield
for chatType, dist in pairs(packets) do
if chatType == "WHISPER" or chatType == "CHANNEL" then
for target, sdist in pairs(dist) do
yield(sdist, chatType, target)
_Recycle(wipe(sdist))
end
else
yield(dist, chatType)
end
_Recycle(wipe(dist))
end
_Recycle(wipe(packets))
end
__Service__(true)
function ProcessMessages()
while true do
while _AddonMessageQueue.Count == 0 do Next() end
local packets = _Recycle()
local callbacks = _Recycle()
local total = 0
-- Build the distribution
repeat
local chatType, prefix, message, target, callback = _AddonMessageQueue:Dequeue(5)
if callback then tasks[callback] = true end
local dist = packets[chatType] or _Recycle()
packets[chatType] = dist
local temp = _Recycle()
message = Toolset.tostring(message)
total = total + #message
temp[1] = prefix
temp[2] = message
if target ~= -1 then
local subDist = dist[target] or _Recycle()
dist[target] = subDist
subDist[#subDist + 1] = temp
else
dist[#dist + 1] = temp
end
-- Not too big
if total >= 256 then break end
until _AddonMessageQueue.Count <= 0
-- Send the packets
for dist, chatType, target in iterdist(packets) do
local extra = #SCORPIO_ADDON_PREFIX + (target and #tostring(target) or 0)
local maxlen = 238 - extra
local message = Base64Encode(DeflateEncode(Toolset.tostring(dist)))
local length = #message
local count = ceil(length / maxlen) -- 12bit for header
local header = format("%06X:", random(0xffffff)) .. toString(count) .. ":"
updateAvailable()
-- Send the messages
for i = 1, count do
local msg = header .. toString(i) .. ":" .. message:sub(1 + (i-1) * maxlen, i * maxlen)
local len = #msg + extra
while _Available < len do
local delay = (len - _Available) / MAX_BYTE_SECOND
if delay >= 0.1 then Delay(delay) else Next() end
updateAvailable()
end
C_ChatInfo.SendAddonMessage(SCORPIO_ADDON_PREFIX, msg, chatType, target)
end
end
-- Invoke the callback or resume the coroutine
for callback in pairs(callbacks) do
if type(callback) == "function" then
local ok, err = pcall(callback)
if not ok then geterrorhandler()(err) end
else
resume(callback)
end
end
_Recycle(callbacks)
Next()
end
end
------------------------------------------------------------
-- Message Attribute --
------------------------------------------------------------
--- Register a handler for the prefix message
-- @usage
-- Scorpio "MyAddon" "v1.0.1"
--
-- __Message__()
-- function PLAYER_COOLDOWN_UPDATE(cooldown, sender)
-- print(cooldown.spellid, cooldown.start, cooldown.duration)
-- end
--
-- __Message__ "PLAYER_COOLDOWN_START" "PLAYER_COOLDOWN_UPDATE"
-- function PLAYER_COOLDOWN(cooldown, sender)
-- end
--
-- SendAddonMessage("PLAYER_COOLDOWN_UPDATE", { spellid = 1923, start = GetTime(), duration = 3 })
__Sealed__()
class "__Message__" (function(_ENV)
extend "IAttachAttribute"
function AttachAttribute(self, target, targettype, owner, name, stack)
if Class.IsObjectType(owner, Scorpio) then
if #self > 0 then
for _, evt in ipairs(self) do
_ModulePrefixHandler[evt] = _ModulePrefixHandler[evt] or {}
_ModulePrefixHandler[evt][owner]= target
end
else
_ModulePrefixHandler[name] = _ModulePrefixHandler[name] or {}
_ModulePrefixHandler[name][owner] = target
end
else
error("__Message__ can only be applyed to objects of Scorpio.", stack + 1)
end
end
----------------------------------------------
-- Property --
----------------------------------------------
property "AttributeTarget" { default = AttributeTargets.Function }
----------------------------------------------
-- Constructor --
----------------------------------------------
__Arguments__{ NEString * 0 }
function __new(cls, ...)
return { ... }, true
end
----------------------------------------------
-- Meta-Method --
----------------------------------------------
__Arguments__{ NEString }
function __call(self, other)
tinsert(self, other)
return self
end
end)
------------------------------------------------------------
-- Message Method --
------------------------------------------------------------
__Static__()
function Scorpio.SendAddonMessage(prefix, message, chatType, target, callback)
if not Struct.ValidateValue(NEString, prefix) then error("Usage: SendAddonMessage(prefix, message[, chatType[, target]][, callback]) - The prefix must be a non-empty string", 2) end
if not (message and Struct.ValidateValue(Serialization.Serializable, message)) then error("Usage: SendAddonMessage(prefix, message[, chatType[, target]][, callback]) - The message must be serializable", 2) end
if type(chatType) == "function" or chatType == true then
callback = chatType
chatType = "PARTY"
elseif type(target) == "function" or target == true then
callback = target
target = nil
end
if chatType and not Enum.ValidateValue(ChatType, chatType) then error("Usage: SendAddonMessage(prefix, message[, chatType[, target]][, callback]) - The chatType is not valid", 2) end
if chatType == "WHISPER" and not Struct.ValidateValue(NEString, target) then error("Usage: SendAddonMessage(prefix, message[, chatType[, target]][, callback]) - The target must be target user name", 2) end
if chatType == "CHANNEL" and not Struct.ValidateValue(NaturalNumber, target) then error("Usage: SendAddonMessage(prefix, message[, chatType[, target]][, callback]) - The target must be a channel id", 2) end
if callback and callback ~= true and type(callback) ~= "function" then error("Usage: SendAddonMessage(prefix, message[, chatType[, target]][, callback]) - The callback must be a function", 2) end
local thread = callback == true and running()
chatType = chatType or "PARTY"
if chatType == "WHISPER" or chatType == "CHANNEL" then
_AddonMessageQueue:Enqueue(chatType, prefix, message, target, thread or callback or false)
else
_AddonMessageQueue:Enqueue(chatType, prefix, message, -1, thread or callback or false)
end
-- Waiting the result
if thread then return yield() end
end
__Arguments__{ NEString, (Number + Callable)/nil, Number/nil } __Static__()
function Scorpio.NextMessage(prefix, callback, timeout)
if type(callback) == "number" then callback, timeout = nil, callback end
if callback then return nextMessageCall(prefix, callback, timeout) end
local thread = running()
if not thread then error("Usage: NextMessage(prefix[, callback]) - The NextMessage must be used in a coroutine") end
_NextMessageTasks[prefix] = _NextMessageTasks[prefix] or _Recycle()
if timeout then
local task = _Recycle()
task[1] = prefix
task[2] = thread
tinsert(_NextMessageTasks[prefix], task)
Delay(timeout, releaseNextMessage, task)
else
tinsert(_NextMessageTasks[prefix], thread)
end
return yield()
end