Skip to content

Commit d8256f1

Browse files
kernelsmithkernelsmith
kernelsmith
authored and
kernelsmith
committed
initial commit
A tool and library for converting javascript to a sequence of ()[]{}+! characters
0 parents  commit d8256f1

File tree

4 files changed

+431
-0
lines changed

4 files changed

+431
-0
lines changed

README.md

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
# Hieroglyphy
2+
3+
A tool and library for converting javascript strings, numbers, and scripts to
4+
equivalent sequences of ()[]{}+! characters that run in the browser.
5+
(Currently Internet Explorer is not supported as it does not support accessing
6+
a string as an array of characters)
7+
8+
## Usage and installation
9+
10+
You can use the class inside hieroglyphy.rb or use the provided obfu_ruby.rb script
11+
which calls into hieroglyphy.rb and gives you some inflation statistics etc.
12+
Example usage of obfu_ruby.rb:$ obfu_ruby.rb script test.js
13+
14+
# This ruby code was ported from the resources below
15+
16+
# Copyright (c) <2012> <Patricio Palladino>
17+
# Hieroglyphy, Python port from JavasCript version by <Patricio Palladino>
18+
# alcuadrado@github ~ mattaereal@github
19+
20+
# Permission is hereby granted, free of charge, to any person obtaining a copy of
21+
# this software and associated documentation files (the "Software"), to deal in
22+
# the Software without restriction, including without limitation the rights to
23+
# use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
24+
# the Software, and to permit persons to whom the Software is furnished to do so,
25+
# subject to the following conditions:
26+
27+
# The above copyright notice and this permission notice shall be included in all
28+
# copies or substantial portions of the Software.
29+
30+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
31+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
32+
# FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
33+
# COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
34+
# IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
35+
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

hieroglyphy.rb

+284
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,284 @@
1+
class JSObfu
2+
# This ruby code was ported from the resources below
3+
4+
# Copyright (c) <2012> <Patricio Palladino>
5+
# Hieroglyphy, Python port from JavasCript version by <Patricio Palladino>
6+
# alcuadrado@github ~ mattaereal@github
7+
8+
# Permission is hereby granted, free of charge, to any person obtaining a copy of
9+
# this software and associated documentation files (the "Software"), to deal in
10+
# the Software without restriction, including without limitation the rights to
11+
# use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
12+
# the Software, and to permit persons to whom the Software is furnished to do so,
13+
# subject to the following conditions:
14+
15+
# The above copyright notice and this permission notice shall be included in all
16+
# copies or substantial portions of the Software.
17+
18+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
20+
# FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
21+
# COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
22+
# IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
23+
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24+
25+
def initialize(code,strip_c=true)
26+
@code = code
27+
strip_comments! if strip_c
28+
init_char_set
29+
end
30+
31+
#
32+
# returns the obfuscated javascript
33+
#
34+
def obfuscate
35+
return transform_script(@code)
36+
end
37+
38+
# Testing
39+
def obfu_num
40+
return transform_number @code
41+
end
42+
def obfu_str
43+
return transform_string @code
44+
end
45+
46+
#
47+
# provide a to_s method
48+
#
49+
def to_s
50+
@code
51+
end
52+
#
53+
# Removes comments and blank lines. Call before obfuscation
54+
#
55+
def strip_comments!
56+
@code = strip_comments
57+
end
58+
59+
#
60+
# Removes comments and blank lines. Call before obfuscation
61+
#
62+
def strip_comments
63+
# removes whole lines only, doesn't remove inline comments
64+
code_to_be_obfu = ''
65+
@code.each_line do |line|
66+
if (not line =~ /^\s*\/\// and not line =~ /^\s+$/)
67+
code_to_be_obfu << line
68+
end
69+
end
70+
return code_to_be_obfu
71+
end
72+
73+
protected
74+
75+
#
76+
# get the hex string of +number+ padded by zeroes to reach +digits+ if necessary
77+
#
78+
def get_hex_string (number, digits)
79+
str = ""
80+
begin
81+
str = number.to_s(16)
82+
rescue Exception => e
83+
# for now just re-raise the error, this is here in case we want to do something else instead
84+
raise e
85+
end
86+
return str.rjust(digits,"0")
87+
end
88+
89+
#
90+
# returns an unescape sequence for a given +charcode+ aka a codepoint
91+
#
92+
def get_unescape_sequence(charcode)
93+
return @kw[:unescape] + "(" + transform_string("%" + get_hex_string(charcode, 2)) + ")"
94+
end
95+
96+
#
97+
# returns a hex sequence for a given +charcode+ aka a codepoint
98+
#
99+
def get_hex_sequence(charcode)
100+
return transform_string( "\\x" + get_hex_string(charcode, 2) )
101+
end
102+
103+
#
104+
# returns a unicode sequence for a +charcode+
105+
#
106+
def get_unicode_sequence (charcode)
107+
return transform_string( "\\u" + get_hex_string(charcode, 4) )
108+
end
109+
110+
#
111+
# returns a hieroglyphied (obfuscated) +char+
112+
#
113+
def transform_char(char)
114+
# if we already know about char, just look it up and return it obfu'd.
115+
if @characters.include?(char)
116+
return @characters[char]
117+
end
118+
charcode = char.bytes.first #char[0].ord # this works in Ruby 1.8 and 1.9, nice.
119+
if (char == "\\" or char == "x")
120+
# These chars must be handled apart because the others need them
121+
@characters[char] = get_unescape_sequence(charcode)
122+
return @characters[char]
123+
end
124+
125+
shortest_sequence = get_unicode_sequence(charcode)
126+
127+
# ASCII@characters can be obtained with hexa and unscape sequences
128+
if (charcode < 128)
129+
unescape_sequence = get_unescape_sequence(charcode)
130+
if (shortest_sequence.length > unescape_sequence.length)
131+
shortest_sequence = unescape_sequence
132+
end
133+
134+
hex_sequence = get_hex_sequence(charcode)
135+
if (shortest_sequence.length > hex_sequence.length)
136+
shortest_sequence = hex_sequence
137+
end
138+
end
139+
140+
@characters[char] = shortest_sequence
141+
return shortest_sequence
142+
end
143+
144+
#
145+
# returns a hieroglyphied (obfuscated) string from +str+
146+
#
147+
def transform_string(str)
148+
glyph_str = ""
149+
str.each_char do |c|
150+
glyph_str += "+" unless glyph_str == ""
151+
glpyh_char = transform_char(c)
152+
glyph_str += glpyh_char
153+
end
154+
return glyph_str
155+
end
156+
157+
#
158+
# Returns a hieroglyphied (obfuscated) number +n+
159+
#
160+
def transform_number(n)
161+
return @numbers[n] if n <= 9
162+
return "+(" + transform_string(n.to_s) + ")"
163+
end
164+
165+
#
166+
# Returns a hieroglyphied (obfuscated) script from +src+
167+
#
168+
def transform_script(src)
169+
return (@kw[:function_constructor] + "(" + transform_string(src) + ")()")
170+
end
171+
172+
protected
173+
174+
def init_char_set
175+
@numbers = [
176+
"+[]", # 0
177+
"+!![]",
178+
"!+[]+!![]",
179+
"!+[]+!![]+!![]",
180+
"!+[]+!![]+!![]+!![]",
181+
"!+[]+!![]+!![]+!![]+!![]", # 5
182+
"!+[]+!![]+!![]+!![]+!![]+!![]",
183+
"!+[]+!![]+!![]+!![]+!![]+!![]+!![]",
184+
"!+[]+!![]+!![]+!![]+!![]+!![]+!![]+!![]",
185+
"!+[]+!![]+!![]+!![]+!![]+!![]+!![]+!![]+!![]" # 9
186+
]
187+
@characters = {
188+
"0" => "(" + @numbers[0] + "+[])",
189+
"1" => "(" + @numbers[1] + "+[])",
190+
"2" => "(" + @numbers[2] + "+[])",
191+
"3" => "(" + @numbers[3] + "+[])",
192+
"4" => "(" + @numbers[4] + "+[])",
193+
"5" => "(" + @numbers[5] + "+[])",
194+
"6" => "(" + @numbers[6] + "+[])",
195+
"7" => "(" + @numbers[7] + "+[])",
196+
"8" => "(" + @numbers[8] + "+[])",
197+
"9" => "(" + @numbers[9] + "+[])"
198+
}
199+
200+
# keywords
201+
@kw = {
202+
:_object_Object => '{}+[]',
203+
:_NaN => '+{}+[]',
204+
:_true => '!![]+[]',
205+
:_false => '![]+[]',
206+
:_undefined => '[][+[]]+[]'
207+
}
208+
@characters[" "] = "(" + @kw[:_object_Object] + ")[" + @numbers[7] + "]"
209+
@characters["["] = "(" + @kw[:_object_Object] + ")[" + @numbers[0] + "]"
210+
@characters["]"] = "(" + @kw[:_object_Object] + ")[" + @characters["1"] + "+" + @characters["4"] + "]"
211+
@characters["a"] = "(" + @kw[:_NaN] + ")[" + @numbers[1] + "]"
212+
@characters["b"] = "(" + @kw[:_object_Object] + ")[" + @numbers[2] + "]"
213+
@characters["c"] = "(" + @kw[:_object_Object] + ")[" + @numbers[5] + "]"
214+
@characters["d"] = "(" + @kw[:_undefined] + ")[" + @numbers[2] + "]"
215+
@characters["e"] = "(" + @kw[:_undefined] + ")[" + @numbers[3] + "]"
216+
@characters["f"] = "(" + @kw[:_undefined] + ")[" + @numbers[4] + "]"
217+
@characters["i"] = "(" + @kw[:_undefined] + ")[" + @numbers[5] + "]"
218+
@characters["j"] = "(" + @kw[:_object_Object] + ")[" + @numbers[3] + "]"
219+
@characters["l"] = "(" + @kw[:_false] + ")[" + @numbers[2] + "]"
220+
@characters["n"] = "(" + @kw[:_undefined] + ")[" + @numbers[1] + "]"
221+
@characters["o"] = "(" + @kw[:_object_Object] + ")[" + @numbers[1] + "]"
222+
@characters["r"] = "(" + @kw[:_true] + ")[" + @numbers[1] + "]"
223+
@characters["s"] = "(" + @kw[:_false] + ")[" + @numbers[3] + "]"
224+
@characters["t"] = "(" + @kw[:_true] + ")[" + @numbers[0] + "]"
225+
@characters["u"] = "(" + @kw[:_undefined] + ")[" + @numbers[0] +"]"
226+
@characters["N"] = "(" + @kw[:_NaN] + ")[" + @numbers[0] + "]"
227+
@characters["O"] = "(" + @kw[:_object_Object] + ")[" + @numbers[8] + "]"
228+
@kw[:_Infinity] = "+(" + @numbers[1] + "+" +@characters["e"] + "+" +
229+
@characters["1"] + "+" +@characters["0"] + "+" +@characters["0"] + "+" +@characters["0"] + ")+[]"
230+
@characters["y"] = "(" + @kw[:_Infinity] + ")[" + @numbers[7] + "]"
231+
@characters["I"] = "(" + @kw[:_Infinity] + ")[" + @numbers[0] + "]"
232+
233+
@kw[:_1e100] = "+(" + @numbers[1] + "+" +@characters["e"] + "+" +
234+
@characters["1"] + "+" +@characters["0"] + "+" +@characters["0"] + ")+[]"
235+
@characters["+"] = "(" + @kw[:_1e100] + ")[" + @numbers[2] + "]"
236+
@kw[:function_constructor] = "[][" + transform_string("sort") + "][" + transform_string("constructor") + "]"
237+
238+
# The chars below need target http(s) pages
239+
@kw[:location_string] = "[]+" + transform_script("return location")
240+
@characters["h"] = "(" + @kw[:location_string] + ")" + "[" + @numbers[0] + "]"
241+
@characters["p"] = "(" + @kw[:location_string] + ")" + "[" + @numbers[3] + "]"
242+
243+
@characters["/"] = "(" + @kw[:location_string] + ")" + "[" + @numbers[6] + "]"
244+
245+
@kw[:unescape] = transform_script("return unescape")
246+
@kw[:escape] = transform_script("return escape")
247+
248+
@characters["%"] = @kw[:escape] + "(" + transform_string("[") + ")[" + @numbers[0] + "]"
249+
250+
do_full_debug if false
251+
end
252+
253+
def bug(msg)
254+
$stderr.puts "[*] #{msg}"
255+
end
256+
def do_full_debug
257+
bug "*************************************"
258+
bug "NUMBERS (AS INTEGERS)"
259+
bug "*************************************"
260+
@numbers.each_with_index do |n,idx|
261+
$stderr.puts "NUMBER #{idx.to_s} is #{n.to_s} (length=#{n.length})"
262+
end
263+
bug "*************************************"
264+
bug "CHARACTERS"
265+
bug "*************************************"
266+
keys = @characters.keys
267+
keys.sort.each do |key|
268+
val = @characters[key]
269+
$stderr.puts "CHAR #{key} is #{val.to_s} (length=#{val.length})"
270+
end
271+
bug "*************************************"
272+
bug "KEYWORDS"
273+
bug "*************************************"
274+
keys = []
275+
@kw.keys.each_with_index do |key, idx|# convert keys to strings so they can be sorted
276+
keys[idx] = key.to_s
277+
end
278+
keys.sort.each do |key|
279+
val = @kw[key.to_sym]
280+
$stderr.puts "KEYWORD #{key} is #{val.to_s} (length=#{val.length})"
281+
end
282+
end
283+
end # end class
284+

0 commit comments

Comments
 (0)