installer.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492
  1. import subprocess
  2. import os
  3. import time
  4. class Installer:
  5. ############################################################
  6. # install_software
  7. ############################################################
  8. def install_software(self):
  9. if self.benchmarker.install == 'all' or self.benchmarker.install == 'server':
  10. self.__install_server_software()
  11. if self.benchmarker.install == 'all' or self.benchmarker.install == 'client':
  12. self.__install_client_software()
  13. ############################################################
  14. # End install_software
  15. ############################################################
  16. ############################################################
  17. # __install_server_software
  18. ############################################################
  19. def __install_server_software(self):
  20. #######################################
  21. # Prerequisites
  22. #######################################
  23. self.__run_command("sudo apt-get update", True)
  24. self.__run_command("sudo apt-get upgrade", True)
  25. self.__run_command("sudo apt-get install build-essential libpcre3 libpcre3-dev libpcrecpp0 libssl-dev zlib1g-dev python-software-properties unzip git-core libcurl4-openssl-dev libbz2-dev libmysqlclient-dev mongodb-clients libreadline6-dev libyaml-dev libsqlite3-dev sqlite3 libxml2-dev libxslt-dev libgdbm-dev ncurses-dev automake libffi-dev htop libtool bison libevent-dev libgstreamer-plugins-base0.10-0 libgstreamer0.10-0 liborc-0.4-0 libwxbase2.8-0 libwxgtk2.8-0 libgnutls-dev libjson0-dev libmcrypt-dev libicu-dev cmake gettext curl libpq-dev", True)
  26. self.__run_command("sudo add-apt-repository ppa:ubuntu-toolchain-r/test", True)
  27. self.__run_command("sudo apt-get update", True)
  28. self.__run_command("sudo apt-get install gcc-4.8 g++-4.8", True)
  29. self.__run_command("cp ../config/benchmark_profile ../../.bash_profile")
  30. self.__run_command("sudo sh -c \"echo '* - nofile 16384' >> /etc/security/limits.conf\"")
  31. #######################################
  32. # Languages
  33. #######################################
  34. #
  35. # Dart
  36. #
  37. self.__run_command("curl https://storage.googleapis.com/dart-editor-archive-integration/latest/dartsdk-linux-64.tar.gz | tar xvz")
  38. #
  39. # Erlang
  40. #
  41. self.__run_command("sudo cp ../config/erlang.list /etc/apt/sources.list.d/erlang.list")
  42. self.__run_command("wget -O - http://binaries.erlang-solutions.com/debian/erlang_solutions.asc | sudo apt-key add -")
  43. self.__run_command("sudo apt-get update")
  44. self.__run_command("sudo apt-get install esl-erlang", True)
  45. #
  46. # Python
  47. #
  48. # .profile is not loaded yet. So we should use full path.
  49. pypy_bin = "~/FrameworkBenchmarks/installs/pypy-2.0.2/bin"
  50. python_bin = "~/FrameworkBenchmarks/installs/python-2.7.5/bin"
  51. python3_bin= "~/FrameworkBenchmarks/installs/python-3.3.2/bin"
  52. def easy_install(pkg, two=True, three=False, pypy=False):
  53. cmd = "/easy_install -U '" + pkg + "'"
  54. if two: self.__run_command(python_bin + cmd)
  55. if three: self.__run_command(python3_bin + cmd)
  56. if pypy: self.__run_command(pypy_bin + cmd)
  57. self.__run_command("curl -L http://bitbucket.org/pypy/pypy/downloads/pypy-2.0.2-linux64.tar.bz2 | tar xj")
  58. self.__run_command("curl -L http://www.python.org/ftp/python/2.7.5/Python-2.7.5.tgz | tar xz")
  59. self.__run_command("curl -L http://www.python.org/ftp/python/3.3.2/Python-3.3.2.tar.xz | tar xJ")
  60. self.__run_command("./configure --prefix=$HOME/FrameworkBenchmarks/installs/python-2.7.5 --disable-shared", cwd="Python-2.7.5")
  61. self.__run_command("./configure --prefix=$HOME/FrameworkBenchmarks/installs/python-3.3.2 --disable-shared", cwd="Python-3.3.2")
  62. self.__run_command("make -j", cwd="Python-2.7.5")
  63. self.__run_command("make install", cwd="Python-2.7.5")
  64. self.__run_command("make -j", cwd="Python-3.3.2")
  65. self.__run_command("make install", cwd="Python-3.3.2")
  66. self.__run_command("wget https://bitbucket.org/pypa/setuptools/downloads/ez_setup.py")
  67. self.__run_command(pypy_bin + "/pypy ez_setup.py")
  68. self.__run_command(python_bin + "/python ez_setup.py")
  69. self.__run_command(python3_bin + "/python3 ez_setup.py")
  70. easy_install('pip==1.3.1', two=True, three=True, pypy=True)
  71. easy_install('MySQL-python==1.2.4')
  72. easy_install('PyMySQL==0.5', pypy=True)
  73. easy_install('PyMySQL3==0.5', two=False, three=True)
  74. easy_install('simplejson==3.3.0', two=True, three=True, pypy=False)
  75. easy_install('psycopg2-2.5.1', three=True)
  76. easy_install('ujson==1.33', three=True)
  77. #
  78. # nodejs
  79. #
  80. self.__run_command("curl http://nodejs.org/dist/v0.10.8/node-v0.10.8-linux-x64.tar.gz | tar xvz")
  81. #
  82. # Java
  83. #
  84. self.__run_command("sudo apt-get install openjdk-7-jdk", True)
  85. self.__run_command("sudo apt-get remove --purge openjdk-6-jre openjdk-6-jre-headless", True)
  86. #
  87. # Ruby/JRuby
  88. #
  89. self.__run_command("curl -L get.rvm.io | bash -s head")
  90. self.__run_command("echo rvm_auto_reload_flag=2 >> ~/.rvmrc")
  91. subprocess.call(["bash", "-c", "source ~/.rvm/scripts/'rvm' && rvm install 2.0.0-p0"])
  92. subprocess.call(["bash", "-c", "source ~/.rvm/scripts/'rvm' && rvm 2.0.0-p0 do gem install bundler"])
  93. subprocess.call(["bash", "-c", "source ~/.rvm/scripts/'rvm' && rvm install jruby-1.7.4"])
  94. subprocess.call(["bash", "-c", "source ~/.rvm/scripts/'rvm' && rvm jruby-1.7.4 do gem install bundler"])
  95. # We need a newer version of jruby-rack
  96. self.__run_command("git clone git://github.com/jruby/jruby-rack.git")
  97. subprocess.call(["bash", "-c", "cd installs/jruby-rack && source ~/.rvm/scripts/'rvm' && rvm jruby-1.7.4 do bundle install"])
  98. subprocess.call(["bash", "-c", "cd installs/jruby-rack && source ~/.rvm/scripts/'rvm' && rvm jruby-1.7.4 do jruby -S bundle exec rake clean gem SKIP_SPECS=true"])
  99. subprocess.call(["bash", "-c", "cd installs/jruby-rack/target && source ~/.rvm/scripts/'rvm' && rvm jruby-1.7.4 do gem install jruby-rack-1.2.0.SNAPSHOT.gem"])
  100. #
  101. # go
  102. #
  103. self.__run_command("curl http://go.googlecode.com/files/go1.1.1.linux-amd64.tar.gz | tar xvz")
  104. #
  105. # Perl
  106. #
  107. # Sometimes this HTTP server returns 404, so retry a few times until it works, but don't retry forever
  108. tries = 0
  109. while True:
  110. self.__run_command("curl http://downloads.activestate.com/ActivePerl/releases/5.16.3.1603/ActivePerl-5.16.3.1603-x86_64-linux-glibc-2.3.5-296746.tar.gz | tar xvz");
  111. if os.path.exists(os.path.join('installs', 'ActivePerl-5.16.3.1603-x86_64-linux-glibc-2.3.5-296746')):
  112. break
  113. tries += 1
  114. if tries >= 30:
  115. raise Exception('Could not download ActivePerl after many retries')
  116. time.sleep(5)
  117. self.__run_command("sudo ./install.sh --license-accepted --prefix /opt/ActivePerl-5.16 --no-install-html", cwd="ActivePerl-5.16.3.1603-x86_64-linux-glibc-2.3.5-296746", send_yes=True)
  118. self.__run_command("curl -L http://cpanmin.us | perl - --sudo App::cpanminus")
  119. self.__run_command("cpanm -f -S DBI DBD::mysql Kelp Dancer Mojolicious Kelp::Module::JSON::XS Dancer::Plugin::Database Starman Plack JSON Web::Simple DBD::Pg JSON::XS EV HTTP::Parser::XS Monoceros EV IO::Socket::IP IO::Socket::SSL")
  120. #
  121. # php
  122. #
  123. self.__run_command("wget --trust-server-names http://www.php.net/get/php-5.4.13.tar.gz/from/us1.php.net/mirror")
  124. self.__run_command("tar xvf php-5.4.13.tar.gz")
  125. self.__run_command("./configure --with-pdo-mysql --with-mysql --with-mcrypt --enable-intl --enable-mbstring --enable-fpm --with-fpm-user=www-data --with-fpm-group=www-data --with-openssl", cwd="php-5.4.13")
  126. self.__run_command("make", cwd="php-5.4.13")
  127. self.__run_command("sudo make install", cwd="php-5.4.13")
  128. self.__run_command("printf \"\\n\" | sudo pecl install apc-beta", cwd="php-5.4.13")
  129. self.__run_command("sudo cp ../config/php.ini /usr/local/lib/php.ini")
  130. self.__run_command("sudo cp ../config/php-fpm.conf /usr/local/lib/php-fpm.conf")
  131. self.__run_command("rm php-5.4.13.tar.gz")
  132. # Composer
  133. self.__run_command("curl -sS https://getcomposer.org/installer | php -- --install-dir=bin")
  134. # Phalcon
  135. self.__run_command("git clone git://github.com/phalcon/cphalcon.git")
  136. self.__run_command("sudo ./install", cwd="cphalcon/build")
  137. # YAF
  138. self.__run_command("sudo pecl install yaf")
  139. #
  140. # Haskell
  141. #
  142. self.__run_command("sudo apt-get install ghc cabal-install", True)
  143. #
  144. # RingoJs
  145. #
  146. self.__run_command("wget http://www.ringojs.org/downloads/ringojs_0.9-1_all.deb")
  147. self.__run_command("sudo apt-get install jsvc", True)
  148. self.__run_command("sudo dpkg -i ringojs_0.9-1_all.deb", True)
  149. self.__run_command("rm ringojs_0.9-1_all.deb")
  150. #
  151. # Mono
  152. #
  153. self.__run_command("git clone git://github.com/mono/mono")
  154. self.__run_command("git checkout mono-3.0.10", cwd="mono")
  155. self.__run_command("./autogen.sh --prefix=/usr/local", cwd="mono")
  156. self.__run_command("make get-monolite-latest", cwd="mono")
  157. self.__run_command("make EXTERNAL_MCS=${PWD}/mcs/class/lib/monolite/gmcs.exe", cwd="mono")
  158. self.__run_command("sudo make install", cwd="mono")
  159. self.__run_command("mozroots --import --sync")
  160. self.__run_command("git clone git://github.com/mono/xsp")
  161. self.__run_command("git checkout 3.0", cwd="xsp")
  162. self.__run_command("./autogen.sh --prefix=/usr/local", cwd="xsp")
  163. self.__run_command("make", cwd="xsp")
  164. self.__run_command("sudo make install", cwd="xsp")
  165. #
  166. # Nimrod
  167. #
  168. self.__run_command("wget http://www.nimrod-code.org/download/nimrod_0.9.2.zip")
  169. self.__run_command("unzip nimrod_0.9.2.zip")
  170. self.__run_command("chmod +x build.sh", cwd="nimrod")
  171. self.__run_command("./build.sh", cwd="nimrod")
  172. self.__run_command("chmod +x install.sh", cwd="nimrod")
  173. self.__run_command("sudo ./install.sh /usr/bin", cwd="nimrod")
  174. #######################################
  175. # Webservers
  176. #######################################
  177. #
  178. # Nginx
  179. #
  180. self.__run_command("curl http://nginx.org/download/nginx-1.4.1.tar.gz | tar xvz")
  181. self.__run_command("./configure", cwd="nginx-1.4.1")
  182. self.__run_command("make", cwd="nginx-1.4.1")
  183. self.__run_command("sudo make install", cwd="nginx-1.4.1")
  184. #
  185. # Openresty (nginx with openresty stuff)
  186. #
  187. self.__run_command("curl http://openresty.org/download/ngx_openresty-1.2.7.5.tar.gz | tar xvz")
  188. self.__run_command("./configure --with-luajit", cwd="ngx_openresty-1.2.7.5")
  189. self.__run_command("make", cwd="ngx_openresty-1.2.7.5")
  190. self.__run_command("sudo make install", cwd="ngx_openresty-1.2.7.5")
  191. #
  192. # Gunicorn
  193. #
  194. easy_install('gunicorn==17.5', two=True, three=True, pypy=True)
  195. # meinheld HEAD supports gunicorn worker on Python 3
  196. easy_install('https://github.com/mopemope/meinheld/archive/master.zip',
  197. two=True, three=True, pypy=True)
  198. #
  199. # Resin
  200. #
  201. self.__run_command("sudo cp -r /usr/lib/jvm/java-1.7.0-openjdk-amd64/include /usr/lib/jvm/java-1.7.0-openjdk-amd64/jre/bin/")
  202. self.__run_command("curl http://www.caucho.com/download/resin-4.0.36.tar.gz | tar xz")
  203. self.__run_command("./configure --prefix=`pwd`", cwd="resin-4.0.36")
  204. self.__run_command("make", cwd="resin-4.0.36")
  205. self.__run_command("make install", cwd="resin-4.0.36")
  206. self.__run_command("mv conf/resin.properties conf/resin.properties.orig", cwd="resin-4.0.36")
  207. self.__run_command("cat ../config/resin.properties > resin-4.0.36/conf/resin.properties")
  208. self.__run_command("mv conf/resin.xml conf/resin.xml.orig", cwd="resin-4.0.36")
  209. self.__run_command("cat ../config/resin.xml > resin-4.0.36/conf/resin.xml")
  210. ##############################################################
  211. #
  212. # Frameworks
  213. #
  214. ##############################################################
  215. ##############################
  216. # Tornado
  217. ##############################
  218. easy_install('tornado==3.1', two=True, three=True, pypy=True)
  219. easy_install('motor==0.1.1', two=True, three=True, pypy=True)
  220. easy_install('pymongo==2.5.2', two=True, three=True, pypy=True)
  221. ##############################
  222. # Django
  223. ##############################
  224. easy_install("https://www.djangoproject.com/download/1.6b1/tarball/", two=True, three=True, pypy=True)
  225. ##############################
  226. # Grails
  227. ##############################
  228. self.__run_command("wget http://dist.springframework.org.s3.amazonaws.com/release/GRAILS/grails-2.1.1.zip")
  229. self.__run_command("unzip -o grails-2.1.1.zip")
  230. self.__run_command("rm grails-2.1.1.zip")
  231. ##############################
  232. # Flask
  233. ##############################
  234. easy_install('Werkzeug==0.9.2', two=True, three=True, pypy=True)
  235. easy_install('flask==0.10.1', two=True, three=True, pypy=True)
  236. easy_install('sqlalchemy==0.8.2', two=True, three=True, pypy=True)
  237. easy_install('Jinja2==2.7', two=True, three=True, pypy=True)
  238. # Flask-SQLAlchemy HEAD supports Python 3
  239. easy_install('https://github.com/mitsuhiko/flask-sqlalchemy/archive/master.zip',
  240. two=True, three=True, pypy=True)
  241. ##############################
  242. # Bottle
  243. ##############################
  244. easy_install('bottle==0.11.6', two=True, three=True, pypy=True)
  245. easy_install('bottle-sqlalchemy==0.4', two=True, three=True, pypy=True)
  246. ##############################
  247. # Play 2
  248. ##############################
  249. self.__run_command("wget http://downloads.typesafe.com/play/2.1.2-RC1/play-2.1.2-RC1.zip")
  250. self.__run_command("unzip -o play-2.1.2-RC1.zip")
  251. self.__run_command("rm play-2.1.2-RC1.zip")
  252. ##############################
  253. # Play 1
  254. ##############################
  255. self.__run_command("wget http://downloads.typesafe.com/releases/play-1.2.5.zip")
  256. self.__run_command("unzip -o play-1.2.5.zip")
  257. self.__run_command("rm play-1.2.5.zip")
  258. self.__run_command("mv play-1.2.5/play play-1.2.5/play1")
  259. # siena
  260. self.__run_command("play-1.2.5/play1 install siena", send_yes=True)
  261. ##############################
  262. # TreeFrog Framework
  263. ##############################
  264. self.__run_command("sudo apt-get install qt4-qmake libqt4-dev libqt4-sql-mysql g++", True)
  265. self.__run_command("wget http://downloads.sourceforge.net/project/treefrog/src/treefrog-1.6.tar.gz")
  266. self.__run_command("tar xzf treefrog-1.6.tar.gz")
  267. self.__run_command("rm treefrog-1.6.tar.gz")
  268. self.__run_command("./configure --enable-mongo", cwd="treefrog-1.6")
  269. self.__run_command("make", cwd="treefrog-1.6/src")
  270. self.__run_command("sudo make install", cwd="treefrog-1.6/src")
  271. self.__run_command("make", cwd="treefrog-1.6/tools")
  272. self.__run_command("sudo make install", cwd="treefrog-1.6/tools")
  273. ##############################
  274. # Vert.x
  275. ##############################
  276. self.__run_command("curl http://vert-x.github.io/vertx-downloads/downloads/vert.x-1.3.1.final.tar.gz | tar xvz")
  277. ##############################
  278. # Yesod
  279. ##############################
  280. self.__run_command("cabal update")
  281. self.__run_command("cabal install yesod persistent-mysql")
  282. ##############################
  283. # Jester
  284. ##############################
  285. self.__run_command("git clone git://github.com/dom96/jester.git jester/jester")
  286. ##############################################################
  287. #
  288. # System Tools
  289. #
  290. ##############################################################
  291. ##############################
  292. # Maven
  293. ##############################
  294. # self.__run_command("sudo apt-get install maven2", send_yes=True)
  295. self.__run_command("curl www.us.apache.org/dist/maven/maven-3/3.0.5/binaries/apache-maven-3.0.5-bin.tar.gz | tar xvz")
  296. ##############################
  297. # Leiningen
  298. ##############################
  299. self.__run_command("mkdir -p bin")
  300. self.__run_command("wget https://raw.github.com/technomancy/leiningen/stable/bin/lein")
  301. self.__run_command("mv lein bin/lein")
  302. self.__run_command("chmod +x bin/lein")
  303. ############################################################
  304. # End __install_server_software
  305. ############################################################
  306. ############################################################
  307. # __install_client_software
  308. ############################################################
  309. def __install_client_software(self):
  310. subprocess.call(self.benchmarker.sftp_string(batch_file="config/client_sftp_batch"), shell=True)
  311. remote_script = """
  312. ##############################
  313. # Prerequisites
  314. ##############################
  315. yes | sudo apt-get update
  316. yes | sudo apt-get install build-essential git libev-dev libpq-dev libreadline6-dev postgresql
  317. sudo sh -c "echo '* - nofile 16384' >> /etc/security/limits.conf"
  318. sudo mkdir -p /ssd
  319. sudo mkdir -p /ssd/log
  320. ##############################
  321. # MySQL
  322. ##############################
  323. sudo sh -c "echo mysql-server mysql-server/root_password_again select secret | debconf-set-selections"
  324. sudo sh -c "echo mysql-server mysql-server/root_password select secret | debconf-set-selections"
  325. yes | sudo apt-get install mysql-server
  326. sudo stop mysql
  327. # use the my.cnf file to overwrite /etc/mysql/my.cnf
  328. sudo mv /etc/mysql/my.cnf /etc/mysql/my.cnf.orig
  329. sudo mv my.cnf /etc/mysql/my.cnf
  330. sudo cp -R -p /var/lib/mysql /ssd/
  331. sudo cp -R -p /var/log/mysql /ssd/log
  332. sudo cp usr.sbin.mysqld /etc/apparmor.d/
  333. sudo /etc/init.d/apparmor reload
  334. sudo start mysql
  335. # Insert data
  336. mysql -uroot -psecret < create.sql
  337. ##############################
  338. # Postgres
  339. ##############################
  340. sudo useradd benchmarkdbuser -p benchmarkdbpass
  341. sudo -u postgres psql template1 < create-postgres-database.sql
  342. sudo -u benchmarkdbuser psql hello_world < create-postgres.sql
  343. sudo -u postgres -H /etc/init.d/postgresql stop
  344. sudo mv postgresql.conf /etc/postgresql/9.1/main/postgresql.conf
  345. sudo mv pg_hba.conf /etc/postgresql/9.1/main/pg_hba.conf
  346. sudo cp -R -p /var/lib/postgresql/9.1/main /ssd/postgresql
  347. sudo -u postgres -H /etc/init.d/postgresql start
  348. ##############################
  349. # wrk
  350. ##############################
  351. git clone https://github.com/wg/wrk.git
  352. cd wrk
  353. make
  354. sudo cp wrk /usr/local/bin
  355. cd ~
  356. git clone https://github.com/wg/wrk.git wrk-pipeline
  357. cd wrk-pipeline
  358. git checkout pipeline
  359. make
  360. sudo cp wrk /usr/local/bin/wrk-pipeline
  361. cd ~
  362. ##############################
  363. # MongoDB
  364. ##############################
  365. sudo apt-key adv --keyserver hkp://keyserver.ubuntu.com:80 --recv 7F0CEB10
  366. sudo cp 10gen.list /etc/apt/sources.list.d/10gen.list
  367. sudo apt-get update
  368. yes | sudo apt-get install mongodb-10gen
  369. sudo stop mongodb
  370. sudo mv /etc/mongodb.conf /etc/mongodb.conf.orig
  371. sudo mv mongodb.conf /etc/mongodb.conf
  372. sudo cp -R -p /var/lib/mongodb /ssd/
  373. sudo cp -R -p /var/log/mongodb /ssd/log/
  374. sudo start mongodb
  375. """
  376. p = subprocess.Popen(self.benchmarker.ssh_string.split(" "), stdin=subprocess.PIPE)
  377. p.communicate(remote_script)
  378. ############################################################
  379. # End __parse_results
  380. ############################################################
  381. ############################################################
  382. # __run_command
  383. ############################################################
  384. def __run_command(self, command, send_yes=False, cwd=None):
  385. try:
  386. cwd = os.path.join(self.install_dir, cwd)
  387. except AttributeError:
  388. cwd = self.install_dir
  389. if send_yes:
  390. subprocess.Popen(command, shell=True, stdin=subprocess.PIPE, cwd=cwd).communicate("yes")
  391. else:
  392. subprocess.call(command, shell=True, cwd=cwd)
  393. ############################################################
  394. # End __run_command
  395. ############################################################
  396. ############################################################
  397. # __init__(benchmarker)
  398. ############################################################
  399. def __init__(self, benchmarker):
  400. self.benchmarker = benchmarker
  401. self.install_dir = "installs"
  402. try:
  403. os.mkdir(self.install_dir)
  404. except OSError:
  405. pass
  406. ############################################################
  407. # End __init__
  408. ############################################################