-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathSimc.lua
580 lines (555 loc) · 20.9 KB
/
Simc.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
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
--Only for local testing
player = { token = "player" }
target = { token = "target" }
--
local Simc = { }
local debug_tokens = false
local function split(p,d)
local t, ll
t={}
ll=0
if(#p == 1) then
return {p}
end
while true do
l = string.find(p, d, ll, true)
if l ~= nil then
table.insert(t, string.sub(p,ll,l-1))
ll = l + 1
else
table.insert(t, string.sub(p,ll))
break
end
end
return t
end
local function convert_spell_name(name)
local str = ""
local name = name or "UNKNOWN"
if string.find(name, "_") then
local splits = split(name, "_")
local index = 0
for _, v in next, splits do
if index > 0 then
str = str .. v:gsub("^%l", string.upper)
else
str = str .. v
end
index = index + 1
end
else
str = name
end
return str
end
--Token section
local TOK_UNKNOWN = 0
local TOK_PLUS = 1
local TOK_MINUS = 2
local TOK_MULT = 3
local TOK_DIV = 4
local TOK_ADD = 5
local TOK_SUB = 6
local TOK_AND = 7
local TOK_OR = 8
local TOK_XOR = 9
local TOK_NOT = 10
local TOK_EQ = 11
local TOK_NOTEQ = 12
local TOK_LT = 13
local TOK_LTEQ = 14
local TOK_GT = 15
local TOK_GTEQ = 16
local TOK_LPAR = 17
local TOK_RPAR = 18
local TOK_IN = 19
local TOK_NOTIN = 20
local TOK_NUM = 21
local TOK_STR = 22
local TOK_ABS = 23
local TOK_SPELL_LIST = 24
local TOK_FLOOR = 25
local TOK_CEIL = 26
local current_index
local action
local action_name
local target_unit
local expr_str
local token_str
local prev_token
local function next_token()
local c = string.sub(expr_str, current_index, current_index)
if c == nil then return TOK_UNKNOWN end
if (prev_token == TOK_FLOOR or prev_token == TOK_CEIL) and not c == "(" then
return TOK_UNKNOWN
end
token_str = string.sub(expr_str, current_index, current_index)
current_index = current_index + 1
if c == "@" then return TOK_ABS end
if c == "+" then return TOK_ADD end
if c == "-" and (prev_token == TOK_STR or prev_token == TOK_NUM or prev_token == TOK_RPAR) then
return TOK_SUB
end
if c == "-" and not prev_token == TOK_STR and not prev_token == TOK_NUM and not prev_token == tok_RPAR and not tonumber(string.sub(expr_str, current_index, current_index)) then
return TOK_SUB
end
if c == "*" then return TOK_MULT end
if c == "%" then return TOK_DIV end
if c == "&" then
if string.sub(expr_str, current_index, current_index) == "&" then
current_index = current_index + 1
end
return TOK_AND
end
if c == "|" then
if string.sub(expr_str, current_index, current_index) == "|" then
current_index = current_index + 1
end
return TOK_OR
end
if c == "^" then
if string.sub(expr_str, current_index, current_index) == "^" then
current_index = current_index + 1
end
return TOK_XOR
end
if c == "~" then
if string.sub(expr_str, current_index, current_index) == "~" then
current_index = current_index + 1
end
return TOK_IN
end
if c == "!" then
if string.sub(expr_str, current_index, current_index) == "=" then
current_index = current_index + 1
token_str = token_str .. "="
return TOK_NOTEQ
end
if string.sub(expr_str, current_index, current_index) == "~" then
current_index = current_index + 1
--FIXME: !~ symbol
token_str = token_str .. "~"
return TOK_NOTIN
end
return TOK_NOT
end
if c == "=" then
if string.sub(expr_str, current_index, current_index) == "=" then
current_index = current_index + 1
end
return TOK_EQ
end
if c == "<" then
if string.sub(expr_str, current_index, current_index) == "=" then
current_index = current_index + 1
token_str = token_str .. "="
return TOK_LTEQ
end
return TOK_LT
end
if c == ">" then
if string.sub(expr_str, current_index, current_index) == "=" then
current_index = current_index + 1
token_str = token_str .. "="
return TOK_GTEQ
end
return TOK_GT
end
if c == "(" then return TOK_LPAR end
if c == ")" then return TOK_RPAR end
if c:match("%w") then
c = string.sub(expr_str, current_index, current_index)
while (c:match("%w") or c:match("%d") or c == "_" or c == ".") do
token_str = token_str .. c
current_index = current_index + 1
c = string.sub(expr_str, current_index, current_index)
end
if token_str:match("floor") then
return TOK_FLOOR
elseif token_str:match("ceil") then
return TOK_CEIL
else
return TOK_STR
end
end
if c:match("%d") or c == "-" then
c = string.sub(expr_str, current_index, current_index)
while (c:match("%d") or c == ".") do
token_str = token_str .. c
current_index = current_index + 1
c = string.sub(expr_str, current_index, current_index)
end
return TOK_NUM
end
if debug_tokens then
print("Unexpected token:", c)
end
return TOK_UNKNOWN
end
local custom_actions = {
["cancel_buff"] = true, -- cancels a buff, just like "/cancelaura" would do in game.
["swap_action_list"] = true,
["run_action_list"] = true, -- calls the specified action list.
["call_action_list"] = true, -- calls the specified action list.
["restart_sequence"] = true, -- will restart the specified sequence.
["restore_mana"] = true, -- forcefully and instantly restores mana.
["sequence"] = true, -- sub-actions chains to execute in a given order
["strict_sequence"] = true, -- strict sequences are a breed of sequence, except for they do not need to be reset, and when they are started, they cannot be stopped under normal circumstances.
["snapshot_stats"] = true, -- forces simulationcraft to capture your buffed stats values just before the combat actually begins.
["start_moving"] = true, -- triggers a movement phase, it will end only with stop_moving.
["stop_moving"] = true, -- ends the movement phase.
["use_item"] = true, -- triggers the use of an item.
["use_items"] = true, -- triggers the use of an item.
["wait"] = true, -- orders the application to stop processing the actions list for a given time. Auto-attacks and such will still be performed.
["wait_until_ready"] = true, -- orders the player to stop processing the actions list until some cooldown or dot expires.
["pool_resource"] = true, -- will force the application to stop processing the actions list while the resource is restored.
["variable"] = true,
}
local is_custom_action = false
local tokens_present = false
--Spell data section
local data_map = {
["spell"] = true,
["talent"] = true,
["target"] = true,
["variable"] = true,
["buff"] = true,
["debuff"] = true,
["spell_targets"] = true,
["cast_time"] = true,
["cooldown"] = true,
["gcd"] = true,
--Resources
["focus"] = true,
["fury"] = true,
["holy_power"] = true,
}
local function parse_resource_token(data_type, data_identifier)
if not data_identifier then
--return "player:" .. convert_spell_name(data_type) .. "()"
--return "power." .. convert_spell_name(data_type) .. ".amount()"
return convert_spell_name(data_type)
end
if data_identifier == "max" then
--return "player." .. convert_spell_name(data_type) .. ":Max()"
-- return "power." .. convert_spell_name(data_type) .. ".max()"
return convert_spell_name(data_type) .. "Max"
elseif data_identifier == "time_to_max" then
--return "player." .. convert_spell_name(data_type) .. ":TimeToMax()"
-- return "power." .. convert_spell_name(data_type) .. ".ttm()"
return convert_spell_name(data_type) .. "TTM"
elseif data_identifier == "regen" then
-- return "player." .. convert_spell_name(data_type) .. ":Regen()"
-- return "power." .. convert_spell_name(data_type) .. ".regen()"
return convert_spell_name(data_type) .. "Regen"
elseif data_identifier == "deficit" then
-- return "player." .. convert_spell_name(data_type) .. ":Deficit()"
-- return "power." .. convert_spell_name(data_type) .. ".deficit()"
return convert_spell_name(data_type) .. "Deficit"
end
end
local function parse_expression(name_str)
--print(name_str)
local expr = ""
local splits = split(name_str, ".")
local data_type = splits[1]
if data_type then
local data_type = splits[1]
local data_identifier = splits[2]
local data_arg = splits[3]
local spellName = data_identifier and convert_spell_name(data_identifier) or ""
-- print(tostring(action).." | "..tostring(data_type).." | "..tostring(data_identifier).." | "..tostring(data_arg))
--return converter(action,data_type,data_identifier,data_arg)
if data_type == "action" then
if data_arg == "cost" then
return "cast.cost."..spellName.."()"
elseif data_arg == "in_flight" then
return "cast.last." .. action .. "()"
end
elseif data_type == "active_enemies" then
return "#enemies.yards"
elseif data_type == "race" then
return "race == " .. "\"" .. spellName .. "\""
elseif data_type == "talent" then
if data_arg == "enabled" then
-- return convert_spell_name(data_identifier) .. "()"
return "talent." .. spellName
end
elseif data_type == "target" then
target_unit = target
if data_identifier == "health" and data_arg == "pct" then
return "thp"
else
return parse_expression(name_str:sub(string.len("target") + 2))
end
elseif data_type == "variable" then
return name_str
elseif data_type == "buff" then
if data_arg == "up" then
-- return target_unit.token .. ":" .. "Buff(" .. convert_spell_name(data_identifier) ..")"
return "buff." .. spellName .. ".exists()"
elseif data_arg == "down" then
-- return "not " ..target_unit.token .. ":" .. "Buff(" .. convert_spell_name(data_identifier) ..")"
return "not buff." .. spellName .. ".exists()"
elseif data_arg == "remains" then
-- return target_unit.token .. ":" .. "BuffDuration(" .. convert_spell_name(data_identifier) ..")"
return "buff." .. spellName .. ".remain()"
elseif data_arg == "react" then
-- return target_unit.token .. ":" .. "BuffCount(" .. convert_spell_name(data_identifier) ..")"
return "buff." .. spellName .. ".exists()"
elseif data_arg == "stack" then
-- return target_unit.token .. ":" .. "BuffCount(" .. convert_spell_name(data_identifier) ..")"
return "buff." .. spellName .. ".stack()"
end
return name_str
elseif data_type == "cooldown" then
if data_arg == "remains" then
-- return convert_spell_name(data_identifier) .. ":Cooldown()"
return "cd." .. spellName .. ".remain()"
elseif data_arg == "up" or data_arg == "ready" then
-- return convert_spell_name(data_identifier) .. ":CooldownReady()"
return "not cd." .. spellName .. ".exists()"
elseif data_arg == "charges" then
return "charges." .. spellName .. ".count()"
elseif data_arg == "charges_fractional" then
-- return convert_spell_name(data_identifier) .. ":Charges()"
return "charges." .. spellName .. ".frac()"
end
return name_str
elseif data_type == "charges" then
return "charges." .. action .. ".count()"
elseif data_type == "debuff" or data_type == "dot" then
if data_arg == "up" or data_arg == "ticking" then
-- return target_unit.token .. ":" .. "Debuff(" .. convert_spell_name(data_identifier) ..")"
return "debuff." .. spellName .. ".exists()"
elseif data_arg == "down" then
-- return " not " ..target_unit.token .. ":" .. "Debuff(" .. convert_spell_name(data_identifier) ..")"
return "not debuff." .. spellName .. ".exists()"
elseif data_arg == "remains" then
-- return target_unit.token .. ":" .. "DebuffDuration(" .. convert_spell_name(data_identifier) ..")"
return "debuff." .. spellName .. ".remain()"
elseif data_arg == "react" then
-- return target_unit.token .. ":" .. "DebuffCount(" .. convert_spell_name(data_identifier) ..")"
return "debuff." .. spellName .. ".exists()"
elseif data_arg == "stack" then
-- return target_unit.token .. ":" .. "BuffCount(" .. convert_spell_name(data_identifier) ..")"
return "debuff." .. spellName .. ".stack()"
elseif data_arg == "refreshable" then
return "debuff." .. spellName .. ".refresh()"
end
return name_str
elseif data_type == "refreshable" then
return "debuff." .. action .. ".refresh()"
elseif data_type == "ticking" then
return "debuff." .. action .. ".exists()"
elseif data_type == "remains" then
return "debuff." .. action .. ".remain()"
elseif data_type == "spell_targets" then
-- return "enemies:InRadius(target, 8)"
return "enemies.yards"
elseif data_type == "full_recharge_time" then
-- return action .. ":FullRecharge()"
return "charges." .. action .. ".timeTillFull()"
elseif data_type == "cast_time" then
-- return action .. ":CastTime()"
return "cast.time." .. action .. "()"
elseif data_type == "set_bonus" then
return data_identifier
--Resources
elseif data_type == "focus" or data_type == "fury" or data_type == "holy_power" or data_type == "energy" then
return parse_resource_token(data_type, data_identifier) or name_str
elseif data_type == "gcd" then
if data_identifier == "max" then
-- return "Rotation:GCDMax()"
return "gcdMax"
elseif data_identifier == "remains" then
-- return "Rotation:GCD()"
return "gcd"
end
elseif data_type == "time_to_die" then
-- return "target.ttd"
return "ttd"
elseif data_type == "equipped" then
return "equiped." .. spellName
end
return name_str
else
print("Unknown expression type:", splits[1])
return name_str
end
end
local function parse_tokens(expr)
if not expr then return end
local tokens = { }
local token
expr_str = expr
current_index = 1
prev_token = TOK_UNKNOWN
while (true) do
token = next_token()
if token == TOK_UNKNOWN then break end
table.insert(tokens, {["type"] = token, ["label"] = token_str})
if tokens[#tokens].type == TOK_STR and not tonumber(tokens[#tokens].label) then
target_unit = player
tokens[#tokens].expression = parse_expression(tokens[#tokens].label)
end
prev_token = token
end
return tokens
end
local function parse_action(expr)
local tokens
if expr:find("if=") then
tokens = expr:sub(select(2, expr:find("if=")) + 1)
else
end
local act = split(expr, ",")
--print(act[1], act[2])
if string.find(act[1], "=/") then
action = split(act[1], "/")[2]
elseif string.find(act[1], "=") then
action = split(act[1], "=")[2]
end
if not custom_actions[action] then
tokens_present = true
-- print(action.." | "..tostring(tokens))
if action == "arcane_torrent" then
action = "racial";
if tokens then
tokens = tokens .. "&race.BloodElf"
else
tokens = "race.BloodElf"
end
end
if action == "blood_fury" then
action = "racial";
if tokens then
tokens = tokens .. "&race.Orc"
else
tokens = "race.Orc"
end
end
if action == "berserking" then
action = "racial";
if tokens then
tokens = tokens .. "&race.Troll"
else
tokens = "race.Troll"
end
end
if action == "lights_judgment" then
action = "racial";
if tokens then
tokens = tokens .. "&race.LightforgedDraenei"
else
tokens = "race.LightforgedDraenei"
end
end
if action == "shadowmeld" then
action = "racial";
if tokens then
tokens = tokens .. "&race.NightElf"
else
tokens = "race.NightElf"
end
end
action = convert_spell_name(action)
else
is_custom_action = true
end
if is_custom_action then
--Parse name=
if action == "variable" or action == "call_action_list" or action == "run_action_list" then
tokens_present = true
if action == "call_action_list" or action == "run_action_list" then list_name = "actionList" else list_name = "" end
action_name = list_name .. "_" .. split(expr:sub(select(2, expr:find("name=")) + 1), ",")[1]:gsub("^%l", string.upper)
end
--Parse value=
if action == "variable" then
tokens = split(expr:sub(select(2, expr:find("value=")) + 1), ",")[1]
end
end
--Parse other actions
return tokens
end
function Simc:tokens_to_string(tokens)
local expr = ""
for _, t in next, tokens do
if t.type == TOK_AND then
expr = expr .. " and "
elseif t.type == TOK_EQ then
expr = expr .. "=="
elseif t.type == TOK_NOT then
expr = expr .. "not "
elseif t.type == TOK_NOTEQ then
expr = expr .. "~="
elseif t.type == TOK_OR then
expr = expr .. " or "
elseif t.type == TOK_DIV then
expr = expr .. " / "
elseif t.type == TOK_STR and not tonumber(t.label) then
if t.expression then
expr = expr .. t.expression
else
expr = expr .. "ERROR"
end
else
expr = expr .. t.label
end
if debug_tokens then
print(t.type, t.label)
end
end
return tostring(expr)
end
function Simc:Parse(input)
is_custom_action = false
tokens_present = false
action = nil
action_name = nil
local expr = ""
if input:match("actions") then
local tokens = parse_action(input)
if tokens then
tokens = parse_tokens(tokens)
end
local conditionals = tokens and self:tokens_to_string(tokens) or ""
--Add hostile check here and add target automatically somehow
if not is_custom_action then
-- expr = "if " .. action .. ":CanCast()"
expr = "if cast.able." .. action .. "()"
if action and (tokens or conditionals ~= "") then
local part1 = conditionals:match("(.+)not not")
local part2 = conditionals:match("not not%s+(%S+)")
if part1 and part2 then conditionals = part1 .. part2 end
expr = expr .. " and (" .. conditionals .. ")"
end
-- expr = expr .. " then\n return " .. action .. ":Cast()\nend"
expr = expr .. " then\n if cast." .. action .. "() then return end\nend"
elseif action == "variable" then
if tokens and tokens_present then
expr = "local " .. action_name .. " = " .. conditionals
end
elseif action == "call_action_list" or action == "run_action_list" then
if tokens then
expr = "if " .. conditionals .. " then\n "
end
-- expr = expr .. "if self:" .. action_name .. "() then return true end"
expr = expr .. "if " .. action_name .. "() then return end"
if tokens then
expr = expr .. "\nend"
end
else
expr = "--TODO: parsing " .. action
end
end
return expr
end
local rotation = [[
actions.execute+=/deadly_calm,if=cooldown.bladestorm.remains>6&((cooldown.colossus_smash.remains<2|(talent.warbreaker.enabled&cooldown.warbreaker.remains<2))|(equipped.weight_of_the_earth&cooldown.heroic_leap.remains<2))
]]
local rotation = split(rotation, "\n")
for _, v in next, rotation do
print("--" .. v)
print(Simc:Parse(v))
end