mirror of
https://github.com/kovidgoyal/calibre.git
synced 2025-06-22 15:00:46 -04:00
199 lines
7.7 KiB
Python
199 lines
7.7 KiB
Python
#!/usr/bin/env python2
|
|
# vim:fileencoding=UTF-8:ts=4:sw=4:sta:et:sts=4:ai
|
|
from __future__ import (unicode_literals, division, absolute_import,
|
|
print_function)
|
|
|
|
__license__ = 'GPL v3'
|
|
__copyright__ = '2009, Kovid Goyal <kovid@kovidgoyal.net>'
|
|
__docformat__ = 'restructuredtext en'
|
|
|
|
import subprocess, os, time, socket
|
|
|
|
from setup import Command, installer_name
|
|
from setup.build_environment import BUILD_HOST, PROJECT
|
|
|
|
BASE_RSYNC = ['rsync', '-av', '--delete', '--force']
|
|
EXCLUDES = []
|
|
for x in [
|
|
'/src/calibre/plugins', '/manual', '/translations', '/build', '/dist', '/imgsrc', '/format_docs'
|
|
'.bzr', '.git', '.build', '.svn', '*.pyc', '*.pyo', '*.swp', '*.swo',]:
|
|
EXCLUDES.extend(['--exclude', ('/calibre' + x) if x.startswith('/') else x])
|
|
SAFE_EXCLUDES = ['"%s"'%x if '*' in x else x for x in EXCLUDES]
|
|
|
|
def get_rsync_pw():
|
|
return open('/home/kovid/work/env/private/buildbot').read().decode('utf-8').partition(
|
|
':')[-1].strip()
|
|
|
|
def is_vm_running(name):
|
|
qname = '"%s"' % name
|
|
try:
|
|
lines = subprocess.check_output('VBoxManage list runningvms'.split()).decode('utf-8').splitlines()
|
|
except Exception:
|
|
time.sleep(1)
|
|
lines = subprocess.check_output('VBoxManage list runningvms'.split()).decode('utf-8').splitlines()
|
|
for line in lines:
|
|
if line.startswith(qname):
|
|
return True
|
|
return False
|
|
|
|
def is_host_reachable(name, timeout=1):
|
|
try:
|
|
socket.create_connection((name, 22), timeout).close()
|
|
return True
|
|
except:
|
|
return False
|
|
|
|
class Rsync(Command):
|
|
|
|
description = 'Sync source tree from development machine'
|
|
|
|
SYNC_CMD = ' '.join(BASE_RSYNC+SAFE_EXCLUDES+
|
|
['rsync://buildbot@{host}/work/{project}', '..'])
|
|
|
|
def run(self, opts):
|
|
cmd = self.SYNC_CMD.format(host=BUILD_HOST, project=PROJECT)
|
|
env = dict(os.environ)
|
|
env['RSYNC_PASSWORD'] = get_rsync_pw()
|
|
self.info(cmd)
|
|
subprocess.check_call(cmd, shell=True, env=env)
|
|
|
|
def push(host, vmname, available):
|
|
if vmname is None:
|
|
hostname = host.partition(':')[0].partition('@')[-1]
|
|
ok = is_host_reachable(hostname)
|
|
else:
|
|
ok = is_vm_running(vmname)
|
|
if ok:
|
|
available[vmname or host] = True
|
|
rcmd = BASE_RSYNC + EXCLUDES + ['.', host]
|
|
print ('\n\nPushing to:', vmname or host, '\n')
|
|
subprocess.check_call(rcmd, stdout=open(os.devnull, 'wb'))
|
|
|
|
class Push(Command):
|
|
|
|
description = 'Push code to another host'
|
|
|
|
def run(self, opts):
|
|
from threading import Thread
|
|
threads, available = {}, {}
|
|
for host, vmname in {
|
|
r'Owner@winxp:/cygdrive/c/Documents\ and\ Settings/Owner/calibre':'winxp',
|
|
'kovid@ox:calibre':None,
|
|
r'kovid@win7:/cygdrive/c/Users/kovid/calibre':'Windows 7',
|
|
'kovid@win7-x64:calibre':'win7-x64',
|
|
'kovid@tiny:calibre':None,
|
|
'kovid@getafix:calibre-src':None,
|
|
}.iteritems():
|
|
threads[vmname or host] = thread = Thread(target=push, args=(host, vmname, available))
|
|
thread.start()
|
|
while threads:
|
|
for name, thread in tuple(threads.iteritems()):
|
|
thread.join(0.01)
|
|
if not thread.is_alive():
|
|
if available.get(name, False):
|
|
print ('\n\n', name, 'done')
|
|
threads.pop(name)
|
|
|
|
|
|
class VMInstaller(Command):
|
|
|
|
INSTALLER_EXT = None
|
|
VM_NAME = None
|
|
FREEZE_COMMAND = None
|
|
FREEZE_TEMPLATE = 'python setup.py {freeze_command}'
|
|
SHUTDOWN_CMD = ['sudo', 'shutdown', '-h', 'now']
|
|
IS_64_BIT = False
|
|
|
|
BUILD_PREFIX = ['#!/bin/sh', 'export CALIBRE_BUILDBOT=1']
|
|
BUILD_RSYNC = ['mkdir -p ~/build/{project}', r'cd ~/build/{project}', Rsync.SYNC_CMD]
|
|
BUILD_CLEAN = ['rm -rf dist/* build/* src/calibre/plugins/*']
|
|
BUILD_BUILD = ['python setup.py build',]
|
|
FORCE_SHUTDOWN = 0 # number of seconds to wait before doing a forced power off (0 means disabled)
|
|
|
|
def add_options(self, parser):
|
|
if not parser.has_option('--dont-shutdown'):
|
|
parser.add_option('-s', '--dont-shutdown', default=False,
|
|
action='store_true', help='Dont shutdown the VM after building')
|
|
if not parser.has_option('--dont-strip'):
|
|
parser.add_option('-x', '--dont-strip', default=False,
|
|
action='store_true', help='Dont strip the generated binaries')
|
|
|
|
def get_build_script(self):
|
|
rs = ['export RSYNC_PASSWORD=%s'%get_rsync_pw()]
|
|
ans = '\n'.join(self.BUILD_PREFIX + rs)+'\n\n'
|
|
ans += ' && \\\n'.join(self.BUILD_RSYNC) + ' && \\\n'
|
|
ans += ' && \\\n'.join(self.BUILD_CLEAN) + ' && \\\n'
|
|
ans += ' && \\\n'.join(self.BUILD_BUILD) + ' && \\\n'
|
|
ans += self.FREEZE_TEMPLATE.format(freeze_command=self.FREEZE_COMMAND) + '\n'
|
|
ans = ans.format(project=PROJECT, host=BUILD_HOST)
|
|
return ans
|
|
|
|
def run_vm(self):
|
|
if is_vm_running(self.VM_NAME):
|
|
return True
|
|
self.__p = subprocess.Popen(("VBoxManage startvm %s --type gui" % self.VM_NAME).split())
|
|
return False
|
|
|
|
def start_vm(self, sleep=75):
|
|
ssh_host = self.VM_NAME
|
|
already_running = self.run_vm()
|
|
if not already_running:
|
|
time.sleep(2)
|
|
print ('Waiting for SSH server to start')
|
|
while not is_host_reachable(ssh_host, timeout=1):
|
|
time.sleep(0.1)
|
|
|
|
def run_vm_builder(self):
|
|
ssh_host = self.VM_NAME
|
|
build_script = self.get_build_script()
|
|
build_script = build_script.encode('utf-8') if isinstance(build_script, unicode) else build_script
|
|
print ('Running VM builder')
|
|
p = subprocess.Popen(['ssh', ssh_host, 'bash -s'], stdin=subprocess.PIPE)
|
|
p.stdin.write(build_script), p.stdin.flush(), p.stdin.close()
|
|
# Save the build script on the build machine for convenient manual
|
|
# invocation, if needed
|
|
p2 = subprocess.Popen(['ssh', ssh_host, 'cat > build-calibre'], stdin=subprocess.PIPE)
|
|
p2.stdin.write(build_script), p2.stdin.flush(), p2.stdin.close()
|
|
p2.wait()
|
|
rc = p.wait()
|
|
if rc != 0:
|
|
raise SystemExit('VM builder failed with error code: %s' % rc)
|
|
self.download_installer()
|
|
|
|
def installer(self):
|
|
return installer_name(self.INSTALLER_EXT, self.IS_64_BIT)
|
|
|
|
def run(self, opts):
|
|
if opts.dont_strip:
|
|
self.FREEZE_COMMAND += ' --dont-strip'
|
|
subprocess.call(['chmod', '-R', '+r', 'recipes'])
|
|
start_time = time.time()
|
|
self.start_vm()
|
|
startup_time = time.time() - start_time
|
|
start_time = time.time()
|
|
self.run_vm_builder()
|
|
print ('Startup completed in %d seconds' % round(startup_time))
|
|
secs = time.time() - start_time
|
|
print ('Build completed in %d minutes %d seconds' % (secs // 60, secs % 60))
|
|
if not opts.dont_shutdown:
|
|
print ('Shutting down', self.VM_NAME)
|
|
subprocess.call(['ssh', self.VM_NAME]+self.SHUTDOWN_CMD)
|
|
if self.FORCE_SHUTDOWN:
|
|
while is_host_reachable(self.VM_NAME):
|
|
time.sleep(0.1) # wait for SSH server to shutdown
|
|
time.sleep(self.FORCE_SHUTDOWN)
|
|
subprocess.check_call(('VBoxManage controlvm %s poweroff' % self.VM_NAME).split())
|
|
|
|
def download_installer(self):
|
|
installer = self.installer()
|
|
subprocess.check_call(['scp',
|
|
self.VM_NAME+':build/calibre/'+installer, 'dist'])
|
|
if not os.path.exists(installer):
|
|
raise SystemExit('Failed to download installer: '+installer)
|
|
|
|
def clean(self):
|
|
installer = self.installer()
|
|
if os.path.exists(installer):
|
|
os.remove(installer)
|
|
|