forked from eset/ipyida
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathinstall_from_ida.py
153 lines (133 loc) · 5.4 KB
/
install_from_ida.py
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
# -*- encoding: utf8 -*-
#
# Automate the installation of pip and IPyIDA.
#
# Copyright (c) 2016-2018 ESET
# Author: Marc-Etienne M.Léveillé <[email protected]>
# See LICENSE file for redistribution.
import idaapi
import os
import sys
from contextlib import contextmanager
import tempfile
import subprocess
import fileinput
import pkg_resources
import shutil
if not "IPYIDA_PACKAGE_LOCATION" in dir():
IPYIDA_PACKAGE_LOCATION = "ipyida"
# Fix the sys.exectuable path. It's misleading in two cases:
# - On Windows, it's set to 'idaq.exe'.
# - If a virtualenv is activated with activate_this.py, sys.prefix is changed
# but sys.executable is still set to the original process. pip and packages
# will not install in the virtualenv if we don't set it right.
if not hasattr(sys, 'real_executable'):
sys.real_executable = sys.executable
if sys.platform == 'win32':
sys.executable = os.path.join(sys.prefix, 'Python.exe')
else:
sys.executable = os.path.join(sys.prefix, 'bin', 'python')
if sys.version_info.major >= 3:
# Ready for Python 4?
sys.executable += str(sys.version_info.major)
# IDA Python sets sys.stdout to a file-like object IDAPythonStdOut. It doesn't
# have things like fileno, close, etc. This helper uses a file and redirect the
# content back to IDA's stdout.
@contextmanager
def temp_file_as_stdout():
ida_stdout = sys.stdout
try:
with tempfile.TemporaryFile() as f:
sys.stdout = f
yield
f.seek(0)
ida_stdout.write(f.read())
finally:
sys.stdout = ida_stdout
try:
import pip
print("[+] Using already installed pip (version {:s})".format(pip.__version__))
except ImportError:
print("[+] Installing pip")
# pip is built-in Python 3 since 3.4, so we assume Python 2 here
# We won't support Python >= 3.0 and < 3.4 unless there's a high demand
import urllib2
if sys.hexversion < 0x02070900:
# There are SSL problems with Python version < 2.7.9
# See https://github.com/eset/ipyida/issues/11
print("[-] IPyIDA installer requires Python 2.7.9 or newer")
raise Exception("Python >= 2.7.9 required")
get_pip = urllib2.urlopen("https://bootstrap.pypa.io/get-pip.py").read()
with temp_file_as_stdout():
p = subprocess.Popen(
sys.executable,
stdin=subprocess.PIPE,
stdout=sys.stdout,
stderr=sys.stdout
)
p.communicate(get_pip)
try:
import pip
except:
print("[-] Could not install pip.")
raise
def pip_install(package, extra_args=[]):
pip_install_cmd = [ sys.executable, "-m", "pip", "install", "--upgrade" ]
with temp_file_as_stdout():
p = subprocess.Popen(
pip_install_cmd + extra_args + [ package ],
stdin=subprocess.PIPE,
stdout=sys.stdout,
stderr=sys.stdout
)
ret = p.wait()
return ret
if pip_install(IPYIDA_PACKAGE_LOCATION) != 0:
print("[.] ipyida system-wide package installation failed, trying user install")
if pip_install(IPYIDA_PACKAGE_LOCATION, [ "--user" ]) != 0:
raise Exception("ipyida package installation failed")
else:
# If no packages were installed in user site-packages, the path may
# not be in sys.path in current Python process. Importing ipyida will
# fail until Python (or IDA) is restarted. We can "refresh" sys.path
# using site.main().
import site
if site.getusersitepackages() not in sys.path:
site.main()
if not os.path.exists(idaapi.get_user_idadir()):
os.makedirs(idaapi.get_user_idadir(), 0o755)
ida_python_rc_path = os.path.join(idaapi.get_user_idadir(), "idapythonrc.py")
rc_file_content = ""
if os.path.exists(ida_python_rc_path):
with open(ida_python_rc_path, "r") as rc:
rc_file_content = rc.read()
if "# BEGIN IPyIDA loading" in rc_file_content:
print("[.] Old IPyIDA loading script present in idapythonrc.py. Removing.")
in_ipyida_block = False
for line in fileinput.input(ida_python_rc_path, inplace=1, backup='.ipyida_old'):
if line.startswith("# BEGIN IPyIDA loading code"):
in_ipyida_block = True
elif line.startswith("# END IPyIDA loading code"):
in_ipyida_block = False
elif not in_ipyida_block:
sys.stdout.write(line)
ipyida_stub_target_path = os.path.join(idaapi.get_user_idadir(), "plugins", "ipyida.py")
if not os.path.exists(os.path.dirname(ipyida_stub_target_path)):
os.makedirs(os.path.dirname(ipyida_stub_target_path), 0o755)
# Make sure ipyida module is not the ipyida.py in the plugins folder, otherwise
# pkg_resources will try to get file from there. This happends when package is
# uninstalled, but ipyida.py is still in the plugin folder.
if 'ipyida' in sys.modules:
del sys.modules['ipyida']
shutil.copyfile(
pkg_resources.resource_filename("ipyida", "ipyida_plugin_stub.py"),
ipyida_stub_target_path
)
print("[+] ipyida.py added to user plugins")
idaapi.load_plugin(ipyida_stub_target_path)
_ida_version = pkg_resources.parse_version(idaapi.get_kernel_version())
if os.name == 'nt' and _ida_version < pkg_resources.parse_version("7.4"):
# No party for Windows with old IDA
print("[+] IPyIDA Installation successful. Use <Shift+.> to open the console.")
else:
print("[🍺] IPyIDA Installation successful. Use <Shift+.> to open the console.")