Developer Stuff - Hekili/hekili GitHub Wiki
page to put things and stuff
Profiling Code
If you want generic plug-and-play, you need from below:
- The macros
- Updated ProfileCPU function:
function Hekili:ProfileCPU( name, func )
- Data storage function:
function Hekili:AddProfileData( name, timeSpent )
- Printer function:
function Hekili:PrintProfileSummary()
- Storage cleanup:
function Hekili:ClearProfileData()
Anything else is optional
In-Game Macros
Print the profiles after combat
/run Hekili:PrintProfileSummary()
/run Hekili:PrintNextPredictionProfile()
Clear the data after printing
/run Hekili:ClearProfileData()
/run Hekili:ResetNextPredictionProfile()
Hekili.lua
ProfileCPU
function
Updated This is the function that allows you to write stuff like this to have a function added to the profiler
Hekili:ProfileCPU( "QueueEvent", state.QueueEvent )
Hekili:ProfileCPU( "ThreadedUpdate", Hekili.Update )
Hekili:ProfileCPU( "GetNextPrediction", Hekili.GetNextPrediction )
Theoretically, you can add any function you want.
local cpuProfileDB = {}
Hekili.CPUProfileResults = {}
function Hekili:ProfileCPU(name, func)
-- Store the original function for reference
print("Profiling function:", name)
cpuProfileDB[name] = func
-- Initialize profiling data if it doesn’t already exist
if not self.CPUProfileResults[name] then
self.CPUProfileResults[name] = {
totalCPUTime = 0,
calls = 0,
maxCPU = 0,
minCPU = math.huge
}
end
-- Redefine the function with a profiling wrapper
self[name] = function(...)
local startCPU = debugprofilestop()
local result = { func(...) }
local endCPU = debugprofilestop()
-- Calculate CPU time for this call
local cpuTime = endCPU - startCPU
-- Ensure profileData is available
local profileData = self.CPUProfileResults[name]
if profileData then
profileData.totalCPUTime = profileData.totalCPUTime + cpuTime
profileData.calls = profileData.calls + 1
profileData.maxCPU = math.max(profileData.maxCPU, cpuTime)
profileData.minCPU = math.min(profileData.minCPU, cpuTime)
else
self:Debug("Warning: profileData is nil for function " .. name)
end
-- Return the original function's results
return unpack(result)
end
end
Hekili:AddProfileData( name, timeSpent )
- Function for storing data over time (live profiling), scalable
function Hekili:AddProfileData( name, timeSpent )
self.ProfileResults = self.ProfileResults or {}
if not self.ProfileResults[ name ] then
self.ProfileResults[ name ] = { totalTime = 0, count = 0, max = 0, min = math.huge }
end
local result = self.ProfileResults[ name ]
result.totalTime = result.totalTime + timeSpent
result.count = result.count + 1
result.max = math.max( result.max, timeSpent )
result.min = math.min( result.min, timeSpent )
end
Hekili:PrintProfileSummary()
- Print Function for the profiler
function Hekili:PrintProfileSummary()
-- Ensure there is profiling data to print
if not self.ProfileResults or next(self.ProfileResults) == nil then
print("No profiling data recorded.")
return
end
-- Prepare a list of keys sorted by total time in descending order
local sortedEntries = {}
for name, data in pairs(self.ProfileResults) do
table.insert(sortedEntries, { name = name, data = data })
end
-- Sort by totalTime in descending order
table.sort(sortedEntries, function(a, b)
return a.data.totalTime > b.data.totalTime
end)
-- General profiling summary
print("\nProfiling Summary:")
print(string.format("%-25s | %12s | %8s | %10s | %10s | %10s", "Function Name", "Total Time", "Calls", "Avg Time", "Max Time", "Min Time"))
print(string.rep("-", 85))
for _, entry in ipairs(sortedEntries) do
local name, data = entry.name, entry.data
local avg = data.totalTime / data.count
print(string.format("%-25s | %10.3f ms | %6d | %8.3f ms | %8.3f ms | %8.3f ms",
name, data.totalTime, data.count, avg, data.max, data.min))
end
-- Specific section for entries containing "reset" or "Reset"
print("\nReset Profiling Summary:")
print(string.format("%-25s | %12s | %8s | %10s | %10s | %10s", "Reset Entry", "Total Time", "Calls", "Avg Time", "Max Time", "Min Time"))
print(string.rep("-", 85))
for _, entry in ipairs(sortedEntries) do
local name, data = entry.name, entry.data
if name:match("[rR]eset") then -- Match any entry containing "reset" or "Reset"
local avg = data.totalTime / data.count
print(string.format("%-25s | %10.3f ms | %6d | %8.3f ms | %8.3f ms | %8.3f ms",
name, data.totalTime, data.count, avg, data.max, data.min))
end
end
-- Check for CPU profiling data
if Hekili.CPUProfileResults then
print("\nCPU Profile Summary:")
-- Sort CPU profiling data by total CPU time in descending order
local cpuSortedEntries = {}
for funcName, data in pairs(Hekili.CPUProfileResults) do
table.insert(cpuSortedEntries, { funcName = funcName, data = data })
end
table.sort(cpuSortedEntries, function(a, b)
return a.data.totalCPUTime > b.data.totalCPUTime
end)
-- Print all entries where the call count is greater than 0
for i = 1, #cpuSortedEntries do
local entry = cpuSortedEntries[i]
local funcName = entry.funcName
local data = entry.data
if data.calls > 0 then -- Only print entries with calls > 0
local avgCPU = data.totalCPUTime / data.calls
-- Top row: function name
print(string.format("Function: %-22s", funcName))
-- Second row: table format for profiling data
print(string.format("Calls | Total CPU Time | Avg CPU | Max CPU | Min CPU"))
print(string.format(" %6d | %14.3f ms | %9.3f ms | %9.3f ms | %9.3f ms",
data.calls, data.totalCPUTime, avgCPU, data.maxCPU, data.minCPU))
print(string.rep("-", 60)) -- Separator line for better readability
end
end
else
print("No CPU profiling data recorded.")
end
-- Add engine performance data if available
if Hekili.Engine and Hekili.Engine.threadUpdates then
local engineData = Hekili.Engine.threadUpdates
print("\nEngine Thread Performance Summary:")
print(string.format("%-25s | %-4s | %-4s", "Metric", "Mean", "Peak"))
print(string.rep("-", 55)) -- Reduced width for separator line
print(string.format("%-18s | %7.3f ms | %7.3f ms", "Mean Clock Time", engineData.meanClockTime or 0, engineData.peakClockTime or 0))
print(string.format("%-18s | %7.3f ms | %7.3f ms", "Mean Work Time", engineData.meanWorkTime or 0, engineData.peakWorkTime or 0))
print(string.format("%-20s | %7.3f | %7.3f", "Mean Frames", engineData.meanFrames or 0, engineData.peakFrames or 0))
print(string.format("%-18s | %7.3f ms | %7.3f ms", "Total Wasted Time", engineData.totalWasted or 0, engineData.peakWasted or 0))
print(string.format("%-14s | %7.3f ms", "Mean Wasted Time", engineData.meanWasted or 0))
else
print("No engine performance data recorded.")
end
end
Hekili:ClearProfileData()
- Clear data for print function
-- Function to clear all profiling data
function Hekili:ClearProfileData()
-- Ensure the ProfileResults and CPUProfileResults tables exist
self.ProfileResults = self.ProfileResults or {}
self.CPUProfileResults = self.CPUProfileResults or {}
-- Wipe the ProfileResults table
wipe(self.ProfileResults)
print("Profiling data cleared.")
-- Wipe the CPUProfileResults table
wipe(self.CPUProfileResults)
print("CPU profiling data cleared.")
-- Clear the engine performance data if available
if Hekili.Engine and Hekili.Engine.threadUpdates then
Hekili.Engine.threadUpdates = {
meanClockTime = 0,
meanWorkTime = 0,
meanFrames = 0,
meanWasted = 0,
firstUpdate = 0,
updates = 0,
updatesPerSec = 0,
peakClockTime = 0,
peakWorkTime = 0,
peakFrames = 0,
peakWasted = 0,
totalWasted = 0
}
print("Engine performance data cleared.")
else
print("No engine performance data to clear.")
end
end
Scripts.lua
Updated CheckScript with profiling for individual APL entries
function scripts:CheckScript(scriptID, action, elem)
-- Capture the action we're checking.
local prev_action = state.this_action
if action then state.this_action = action end
-- Retrieve the script from the database.
local script = self.DB[scriptID]
if not script then
state.this_action = prev_action
return false
end
-- Profiling for the entire CheckScript function.
local startCheckScript
if InCombatLockdown() then
startCheckScript = debugprofilestop()
end
-- Initialize the result variables.
local result, errorMessage
if not elem then
-- Profiling for general Conditions.
local startConditions, timeSpentConditions
if InCombatLockdown() then
startConditions = debugprofilestop()
end
if script.Error then
result = false
errorMessage = script.Error
elseif not script.Conditions then
result = true
else
-- Error handling for Conditions to catch potential issues.
local success, conditionsResult = pcall(script.Conditions)
if success then
result = conditionsResult
else
result = false
errorMessage = "Error in Conditions: " .. conditionsResult
end
end
-- Record time spent on Conditions evaluation.
if InCombatLockdown() and startConditions then
timeSpentConditions = debugprofilestop() - startConditions
Hekili:AddProfileData("CheckScript:Conditions:" .. scriptID, timeSpentConditions)
end
else
-- Profiling for Modifier checks.
local startModifier, timeSpentModifier
if InCombatLockdown() then
startModifier = debugprofilestop()
end
if not script.Modifiers[elem] then
result = nil
errorMessage = elem .. " not set."
else
local success, value = pcall(script.Modifiers[elem])
if success then
result = value
else
result = false
errorMessage = "Error in Modifier " .. elem .. ": " .. value
end
end
-- Record time spent on the Modifier evaluation.
if InCombatLockdown() and startModifier then
timeSpentModifier = debugprofilestop() - startModifier
Hekili:AddProfileData("CheckScript:Modifier:" .. scriptID .. ":" .. elem, timeSpentModifier)
end
end
-- Restore the previous action state.
state.this_action = prev_action
-- Record the total time for CheckScript if applicable.
if InCombatLockdown() and startCheckScript then
local timeSpentTotal = debugprofilestop() - startCheckScript
Hekili:AddProfileData("CheckScript:Total:" .. scriptID, timeSpentTotal)
end
-- Return the result with potential error message.
return result, errorMessage
end
GetModifiers
with profiling
function scripts:GetModifiers(scriptID, out)
print("GetModifiers called for scriptID:", scriptID) -- Debug print
-- Start profiling for the whole GetModifiers function
local startTotal
if InCombatLockdown() then
print("In combat, starting profiling for GetModifiers") -- Debug print
startTotal = debugprofilestop()
end
out = out or {}
local script = self.DB[scriptID]
if not script then
print("No script found for scriptID:", scriptID) -- Debug print
-- End profiling if exiting early
if InCombatLockdown() and startTotal then
local endTotal = debugprofilestop() - startTotal
Hekili:AddProfileData("GetModifiers:" .. scriptID, endTotal)
end
return out
end
for k, v in pairs(script.Modifiers) do
local start, timeSpent
if InCombatLockdown() then
start = debugprofilestop()
print("Profiling modifier:", k) -- Debug print
end
local success, value = pcall(v)
if success then
out[k] = value
else
print("Failed to execute modifier:", k) -- Debug print for failure
end
-- Record profiling data for each modifier
if InCombatLockdown() and start then
timeSpent = debugprofilestop() - start
print("Time spent on modifier:", k, "=", timeSpent) -- Debug print
Hekili:AddProfileData("Modifier:" .. k, timeSpent)
end
end
-- End profiling for the entire function
if InCombatLockdown() and startTotal then
local endTotal = debugprofilestop() - startTotal
Hekili:AddProfileData("GetModifiers:" .. scriptID, endTotal)
print("Total time spent on GetModifiers:", endTotal) -- Debug print
end
return out
end
GetConditionsAndValues
with profiling
function scripts:GetConditionsAndValues(scriptID, listName, actID, recheck)
-- Skip if snapshot or debug is not enabled
if troubleshootingSnapshotTimes or not Hekili.ActiveDebug then return "[no data]" end
-- Adjust scriptID if listName and actID are provided
if listName and actID then
scriptID = scriptID .. ":" .. listName .. ":" .. actID
end
-- Start profiling for the whole GetConditionsAndValues function
local startTotal = debugprofilestop()
-- Retrieve the script from the database
local script = self.DB[scriptID]
-- Check if this is a recheck and profile accordingly
if recheck then
local startRecheck = debugprofilestop()
local result = embedConditionsAndValues(script.RecheckScript, script.RecheckElements)
local endRecheck = debugprofilestop() - startRecheck
Hekili:AddProfileData("RecheckConditions:" .. scriptID, endRecheck)
-- End profiling for total function and return
local endTotal = debugprofilestop() - startTotal
Hekili:AddProfileData("GetConditionsAndValues:" .. scriptID, endTotal)
return result
end
-- Profile if there is a custom Print function
if script.Print then
local startPrint = debugprofilestop()
local result = script.Print()
local endPrint = debugprofilestop() - startPrint
Hekili:AddProfileData("PrintFunction:" .. scriptID, endPrint)
-- End profiling for total function and return
local endTotal = debugprofilestop() - startTotal
Hekili:AddProfileData("GetConditionsAndValues:" .. scriptID, endTotal)
return result
end
-- Profile standard condition evaluation with embedConditionsAndValues
local startEmbed = debugprofilestop()
local result = embedConditionsAndValues(script.SimC, script.Elements)
local endEmbed = debugprofilestop() - startEmbed
Hekili:AddProfileData("ConditionEvaluation:" .. scriptID, endEmbed)
-- End profiling for total function
local endTotal = debugprofilestop() - startTotal
Hekili:AddProfileData("GetConditionsAndValues:" .. scriptID, endTotal)
return result
end
Projects
Track refreshable status of debuffs in real-time across all enemies
Has a lot of potential, needs work.
Targets.lua
ns.refreshableAuraCount = function(auraName)
local startTime = debugprofilestop() -- Start timing
local aura = class.auras[auraName]
if not aura then
return 0
end
local auraID = aura.id
-- Handle dynamic duration (if duration is a function, evaluate it)
local duration = type(aura.duration) == "function" and aura.duration() or aura.duration or 30
local refreshThreshold = 0.3 * duration -- Calculate threshold as 30% of the duration
local refreshableCount = 0
-- Iterate over each nameplate unit (up to 40)
for i = 1, 40 do
local unit = "nameplate" .. i
if not UnitExists(unit) then break end -- Stop if no more nameplates are present
local auraFound = false
-- Check each debuff slot on the unit to find the specific aura
for j = 1, 40 do
local auraData = C_UnitAuras.GetAuraDataBySlot(unit, j)
if auraData and auraData.spellId == auraID then
auraFound = true
local currentTime = GetTime()
local expirationTime = auraData.expirationTime or 0
local remainingTime = expirationTime - currentTime
if remainingTime <= refreshThreshold then
refreshableCount = refreshableCount + 1
end
break
end
end
if not auraFound then
-- If the aura is not found on the unit, consider it refreshable.
refreshableCount = refreshableCount + 1
end
end
-- Calculate elapsed time and add it to the profiling data
local timeSpent = debugprofilestop() - startTime
Hekili:AddProfileData("Refreshable", timeSpent)
return refreshableCount
end
SPECFILE.lua
-- Count of refreshable Garrotes
spec:RegisterStateExpr( "refreshable_garrotes", function ()
return ns.refreshableAuraCount("garrote")
end )
-- Count of refreshable Ruptures
spec:RegisterStateExpr( "refreshable_ruptures", function ()
return ns.refreshableAuraCount("rupture")
end )