Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
P
pyrasite
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Kirill Smelkov
pyrasite
Commits
de4de5a9
Commit
de4de5a9
authored
Mar 22, 2012
by
Luke Macken
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Code injection test improvements. Increase GIL contention.
parent
13d50a75
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
78 additions
and
55 deletions
+78
-55
tests/test_code_injection.py
tests/test_code_injection.py
+78
-55
No files found.
tests/test_code_injection.py
View file @
de4de5a9
...
...
@@ -13,12 +13,14 @@
# You should have received a copy of the GNU General Public License
# along with pyrasite. If not, see <http://www.gnu.org/licenses/>.
#
# Copyright (C) 2011, 2012 Red Hat, Inc.
# Copyright (C) 2011, 2012 Red Hat, Inc.
, Luke Macken <lmacken@redhat.com>
import
os
import
sys
import
time
import
glob
import
signal
import
atexit
import
unittest
import
textwrap
import
tempfile
...
...
@@ -26,62 +28,81 @@ import subprocess
import
pyrasite
subprocesse
s
=
[]
default_p
ayload
=
stop_payload
=
None
pid
s
=
[]
default_p
rogram
=
stop_program
=
None
def
teardown
():
os
.
unlink
(
default_p
ayload
)
os
.
unlink
(
stop_p
ayload
)
for
p
in
subprocesse
s
:
def
cleanup
():
os
.
unlink
(
default_p
rogram
)
os
.
unlink
(
stop_p
rogram
)
for
p
id
in
pid
s
:
try
:
p
.
kill
(
)
os
.
kill
(
pid
,
signal
.
SIGTERM
)
except
:
pass
atexit
.
register
(
cleanup
)
class
TestCodeInjection
(
unittest
.
TestCase
):
def
__init__
(
self
,
*
args
,
**
kw
):
super
(
TestCodeInjection
,
self
).
__init__
(
*
args
,
**
kw
)
global
default_p
ayload
,
stop_payload
self
.
stop_p
ayload
=
default_payload
=
self
.
generate_payload
_stopper
()
self
.
default_p
ayload
=
stop_payload
=
self
.
generate_payload
()
global
default_p
rogram
,
stop_program
self
.
stop_p
rogram
=
stop_program
=
self
.
generate_program
_stopper
()
self
.
default_p
rogram
=
default_program
=
self
.
generate_program
()
def
generate_p
ayload
(
self
,
threads
=
1
):
def
generate_p
rogram
(
self
,
cpu_threads
=
1
,
io_
threads
=
1
):
(
fd
,
filename
)
=
tempfile
.
mkstemp
()
tmp
=
os
.
fdopen
(
fd
,
'w'
)
script
=
textwrap
.
dedent
(
"""
import time, threading
, random
global running
import time, threading
from random import SystemRandom as random
running = True
def snooze():
global running
def cpu_bound():
i = 2
y = 0
def fib(n):
return fib(n - 1) + fib(n - 2)
while running:
time.sleep(random.random())
y += fib(i)
i += 1
def io_bound():
while running:
random.random()
"""
)
for
t
in
range
(
threads
):
script
+=
"threading.Thread(target=snooze).start()
\
n
"
# CPU-bound threads
for
t
in
range
(
cpu_threads
):
script
+=
"threading.Thread(target=cpu_bound).start()
\
n
"
# I/O bound (/dev/urandom)
for
t
in
range
(
io_threads
):
script
+=
"threading.Thread(target=io_bound).start()
\
n
"
tmp
.
write
(
script
)
tmp
.
close
()
return
filename
def
generate_p
ayload
_stopper
(
self
):
def
generate_p
rogram
_stopper
(
self
):
(
fd
,
filename
)
=
tempfile
.
mkstemp
()
tmp
=
os
.
fdopen
(
fd
,
'w'
)
tmp
.
write
(
'globals()["running"] = False'
)
tmp
.
close
()
return
filename
def
run_python
(
self
,
payload
,
exe
=
'python'
):
p
=
subprocess
.
Popen
(
'%s %s'
%
(
exe
,
payload
),
def
generate_single_character_printer
(
self
):
(
fd
,
filename
)
=
tempfile
.
mkstemp
()
tmp
=
os
.
fdopen
(
fd
,
'w'
)
tmp
.
write
(
"print('!')"
)
tmp
.
close
()
return
filename
def
run_python
(
self
,
program
,
exe
=
'python'
):
p
=
subprocess
.
Popen
(
'%s %s'
%
(
exe
,
program
),
shell
=
True
,
stdout
=
subprocess
.
PIPE
,
stderr
=
subprocess
.
PIPE
)
subprocesses
.
append
(
p
)
pids
.
append
(
p
.
pid
)
# FIXME: Hack to ensure that python (3, specifically) fully loads
# before we inject code into it. To optimize this, we should
# have the p
ayload
print something out once it's loaded, then
# have the p
rogram
print something out once it's loaded, then
# we should ensure this happens before injecting.
time
.
sleep
(
0.5
)
return
p
...
...
@@ -93,22 +114,47 @@ class TestCodeInjection(unittest.TestCase):
def
interpreters
(
self
):
for
exe
in
glob
.
glob
(
'/usr/bin/python*.*'
):
try
:
minor_ver
=
int
(
exe
.
split
(
'.'
)[
-
1
])
int
(
exe
.
split
(
'.'
)[
-
1
])
except
ValueError
:
continue
# skip python2.7-config, etc
yield
exe
def
test_many_threads_and_many_payloads
(
self
):
payload
=
self
.
generate_payload
(
threads
=
100
)
def
test_injection_into_all_interpreters_with_no_threads
(
self
):
for
exe
in
self
.
interpreters
():
print
(
"sys.executable = %s"
%
sys
.
executable
)
print
(
"injecting into %s"
%
exe
)
p
=
self
.
run_python
(
'-c "import time; time.sleep(2.0)"'
,
exe
=
exe
)
pyrasite
.
inject
(
p
.
pid
,
'pyrasite/payloads/helloworld.py'
,
verbose
=
True
)
stdout
,
stderr
=
p
.
communicate
()
self
.
assert_output_contains
(
stdout
,
stderr
,
'Hello World!'
)
def
test_injecting_into_all_interpreters_with_gil_contention
(
self
):
program
=
self
.
generate_program
(
cpu_threads
=
3
,
io_threads
=
3
)
try
:
for
exe
in
self
.
interpreters
():
print
(
"sys.executable = %s"
%
sys
.
executable
)
print
(
"injecting into %s"
%
exe
)
p
=
self
.
run_python
(
program
,
exe
=
exe
)
pyrasite
.
inject
(
p
.
pid
,
'pyrasite/payloads/helloworld.py'
,
verbose
=
True
)
pyrasite
.
inject
(
p
.
pid
,
self
.
stop_program
,
verbose
=
True
)
stdout
,
stderr
=
p
.
communicate
()
self
.
assert_output_contains
(
stdout
,
stderr
,
'Hello World!'
)
finally
:
os
.
unlink
(
program
)
def
test_many_payloads_into_program_with_many_threads
(
self
):
program
=
self
.
generate_program
(
cpu_threads
=
3
,
io_threads
=
3
)
for
exe
in
self
.
interpreters
():
p
=
self
.
run_python
(
p
ayload
,
exe
=
exe
)
p
=
self
.
run_python
(
p
rogram
,
exe
=
exe
)
total
=
100
for
i
in
range
(
total
):
pyrasite
.
inject
(
p
.
pid
,
'pyrasite/payloads/helloworld.py'
,
verbose
=
True
)
pyrasite
.
inject
(
p
.
pid
,
self
.
stop_p
ayload
,
verbose
=
True
)
pyrasite
.
inject
(
p
.
pid
,
self
.
stop_p
rogram
,
verbose
=
True
)
stdout
,
stderr
=
p
.
communicate
()
count
=
0
...
...
@@ -116,33 +162,10 @@ class TestCodeInjection(unittest.TestCase):
if
line
.
strip
()
==
'Hello World!'
:
count
+=
1
os
.
unlink
(
p
ayload
)
os
.
unlink
(
p
rogram
)
assert
count
==
total
,
"Read %d hello worlds"
%
count
def
test_threadless_injection_into_all_interpreters
(
self
):
for
exe
in
self
.
interpreters
():
print
(
"sys.executable = %s"
%
sys
.
executable
)
print
(
"injecting into %s"
%
exe
)
p
=
self
.
run_python
(
'-c "import time; time.sleep(2.0)"'
,
exe
=
exe
)
pyrasite
.
inject
(
p
.
pid
,
'pyrasite/payloads/helloworld.py'
,
verbose
=
True
)
stdout
,
stderr
=
p
.
communicate
()
self
.
assert_output_contains
(
stdout
,
stderr
,
'Hello World!'
)
def
test_injecting_threads_into_all_interpreters
(
self
):
payload
=
self
.
generate_payload
(
threads
=
10
)
try
:
for
exe
in
self
.
interpreters
():
print
(
"sys.executable = %s"
%
sys
.
executable
)
print
(
"injecting into %s"
%
exe
)
p
=
self
.
run_python
(
payload
,
exe
=
exe
)
pyrasite
.
inject
(
p
.
pid
,
'pyrasite/payloads/helloworld.py'
,
verbose
=
True
)
pyrasite
.
inject
(
p
.
pid
,
self
.
stop_payload
,
verbose
=
True
)
stdout
,
stderr
=
p
.
communicate
()
self
.
assert_output_contains
(
stdout
,
stderr
,
'Hello World!'
)
finally
:
os
.
unlink
(
payload
)
if
__name__
==
'__main__'
:
unittest
.
main
()
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment