Convert lua to make it human-readable and remove obfuscation

Muna

Convert lua to make it human-readable and remove obfuscation. Based on luaj parser and lua formatter.

Before

function L1_1(A0_2, A1_2)
  local L2_2, L3_2
  L2_2 = A1_2.type
  L3_2 = EventType
  L3_2 = L3_2.EVENT_1
  if L2_2 == L3_2 then
    L2_2 = A1_2.param3
    if not (50 < L2_2) then
      goto lbl_11
    end
  end
  L2_2 = false
  do return L2_2 end
  ::lbl_11::
  L2_2 = ScriptLib
  L2_2 = L2_2.call
  L3_2 = A0_2
  L2_2 = L2_2(L3_2)
  if 2 < L2_2 then
    L2_2 = false
    return L2_2
  end
  L2_2 = true
  return L2_2
end
condition_EVENT_1 = L1_1

after

function condition_EVENT_1(context, args)
    if args.type == EventType.EVENT_1 then
        if not (50 < args.param3) then goto lbl_11 end
    end
    do return false end
    ::lbl_11::
    local var1 = ScriptLib.call(context)
    if 2 < var1 then return false end
    return true
end

Support

  • assign, only one arg (don’t support A,B = C)
  • if-else
  • goto/label/block do … end
  • func call/define
  • return

TODO: loop (for – both 2 type) statement

How it works?

It assumed that the var in any assign statement is used only once, so that we can use a map to store the value and simplify almost all assign statement. When it is referred in expression and function arg, we will output its real value. Also, not all assign statement can be simplified due to grammar and idempotent.

The details below show how it handles with the code.

And you may see it will meet some problems when it faces with loop, when the var will be used more than once.

-- input
L2_2 = ScriptLib
L2_2 = L2_2.call
-- output
-- store
L2_2 : ScriptLib.call

We will set L2_2 to ScriptLib.call and won’t output both of them.

-- input
L5_2 = L1_1.challenge_id_no_record
-- output
-- store
L5_2 : L1_1.challenge_id_no_record

Since L1_1 is a global var, we won’t store its value and keep the reference.

-- input
L2_2 = L2_2(L3_2)
-- output
local var1 = ScriptLib.call(context)
-- store
L2_2 : var1

We use a temporal var to hold the result of a func call due to the idempotent.

-- input
L2_2 = {}
L2_2[1] = 1
L2_2[2] = nil
L2_2[3] = 2
...
L4_2 = ScriptLib
L4_2 = L4_2.call
L5_2 = A0_2
L6_2 = {}
L7_2 = L2_2[L3_2]
L6_2.config_id = L7_2
L4_2(L5_2, L6_2)
-- output
local var2 = {1, nil, 2}
ScriptLib.call(context, {config_id = var2[var1]})
-- store
L2_2 : var2
L6_2 : {config_id = var2[var1]}

To avoid grammar error like {1, nil, 2}[var1], we use a temporal var to hold the table when it is indexed.

-- input
function L2_1(A0_2, A1_2)
  ...
end
condition_EVENT1 = L2_1
-- output
function condition_EVENT1(context, args)
    ...
end
-- store
func1 : Function ...
L2_1 : func1
condition_EVENT1 : func1

We use a temporal name funcN to hold the definition of function. When the block’s parser is done, we will start to output the functions & global definitions.

The first arg in func will be replaced with context, and the second one is args.

Can I convert to another language?

Of course. I write it out as lua, but you can customize the output language by yourself.

GitHub

View Github