mirror of
				https://github.com/kovidgoyal/calibre.git
				synced 2025-11-03 19:17:02 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			353 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			353 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
#!/usr/bin/env python
 | 
						|
# vim:fileencoding=UTF-8:ts=4:sw=4:sta:et:sts=4:ai
 | 
						|
 | 
						|
__license__   = 'GPL v3'
 | 
						|
__copyright__ = '2009, Kovid Goyal <kovid@kovidgoyal.net>'
 | 
						|
__docformat__ = 'restructuredtext en'
 | 
						|
 | 
						|
import os, subprocess, hashlib, shutil, glob, stat, sys, time
 | 
						|
from subprocess import check_call
 | 
						|
from tempfile import NamedTemporaryFile, mkdtemp, gettempdir
 | 
						|
from zipfile import ZipFile
 | 
						|
 | 
						|
if __name__ == '__main__':
 | 
						|
    d = os.path.dirname
 | 
						|
    sys.path.insert(0, d(d(os.path.abspath(__file__))))
 | 
						|
 | 
						|
from setup import Command, __version__, installer_name, __appname__
 | 
						|
 | 
						|
PREFIX = "/var/www/calibre-ebook.com"
 | 
						|
DOWNLOADS = PREFIX+"/htdocs/downloads"
 | 
						|
BETAS = DOWNLOADS +'/betas'
 | 
						|
HTML2LRF = "calibre/ebooks/lrf/html/demo"
 | 
						|
TXT2LRF  = "src/calibre/ebooks/lrf/txt/demo"
 | 
						|
STAGING_HOST = 'download.calibre-ebook.com'
 | 
						|
STAGING_USER = 'root'
 | 
						|
STAGING_DIR = '/root/staging'
 | 
						|
 | 
						|
def installers():
 | 
						|
    installers = list(map(installer_name, ('dmg', 'msi', 'txz')))
 | 
						|
    installers.append(installer_name('txz', is64bit=True))
 | 
						|
    installers.append(installer_name('msi', is64bit=True))
 | 
						|
    installers.insert(0, 'dist/%s-%s.tar.xz'%(__appname__, __version__))
 | 
						|
    installers.append('dist/%s-portable-installer-%s.exe'%(__appname__, __version__))
 | 
						|
    return installers
 | 
						|
 | 
						|
def installer_description(fname):
 | 
						|
    if fname.endswith('.tar.xz'):
 | 
						|
        return 'Source code'
 | 
						|
    if fname.endswith('.txz'):
 | 
						|
        bits = '32' if 'i686' in fname else '64'
 | 
						|
        return bits + 'bit Linux binary'
 | 
						|
    if fname.endswith('.msi'):
 | 
						|
        return 'Windows %sinstaller'%('64bit ' if '64bit' in fname else '')
 | 
						|
    if fname.endswith('.dmg'):
 | 
						|
        return 'OS X dmg'
 | 
						|
    if fname.endswith('.exe'):
 | 
						|
        return 'Calibre Portable'
 | 
						|
    return 'Unknown file'
 | 
						|
 | 
						|
def upload_signatures():
 | 
						|
    tdir = mkdtemp()
 | 
						|
    for installer in installers():
 | 
						|
        if not os.path.exists(installer):
 | 
						|
            continue
 | 
						|
        with open(installer, 'rb') as f:
 | 
						|
            raw = f.read()
 | 
						|
        fingerprint = hashlib.sha512(raw).hexdigest()
 | 
						|
        fname = os.path.basename(installer+'.sha512')
 | 
						|
        with open(os.path.join(tdir, fname), 'wb') as f:
 | 
						|
            f.write(fingerprint)
 | 
						|
    check_call('scp %s/*.sha512 divok:%s/signatures/' % (tdir, DOWNLOADS),
 | 
						|
            shell=True)
 | 
						|
    shutil.rmtree(tdir)
 | 
						|
 | 
						|
class ReUpload(Command):  # {{{
 | 
						|
 | 
						|
    description = 'Re-upload any installers present in dist/'
 | 
						|
 | 
						|
    sub_commands = ['upload_installers']
 | 
						|
 | 
						|
    def pre_sub_commands(self, opts):
 | 
						|
        opts.replace = True
 | 
						|
        exists = {x for x in installers() if os.path.exists(x)}
 | 
						|
        if not exists:
 | 
						|
            print ('There appear to be no installers!')
 | 
						|
            raise SystemExit(1)
 | 
						|
 | 
						|
    def run(self, opts):
 | 
						|
        for x in installers():
 | 
						|
            if os.path.exists(x):
 | 
						|
                os.remove(x)
 | 
						|
# }}}
 | 
						|
 | 
						|
# Data {{{
 | 
						|
def get_google_data():
 | 
						|
    with open(os.path.expanduser('~/work/env/private/googlecodecalibre'), 'rb') as f:
 | 
						|
        gc_password, ga_un, pw = f.read().strip().split('|')
 | 
						|
 | 
						|
    return {
 | 
						|
        'username':ga_un, 'password':pw, 'gc_password':gc_password,
 | 
						|
        'path_map_server':'root@kovidgoyal.net',
 | 
						|
        'path_map_location':'/var/www/status.calibre-ebook.com/googlepaths',
 | 
						|
        # If you change this remember to change it in the
 | 
						|
        # status.calibre-ebook.com server as well
 | 
						|
        'project':'calibre-ebook'
 | 
						|
    }
 | 
						|
 | 
						|
def get_github_data():
 | 
						|
    with open(os.path.expanduser('~/work/env/private/github'), 'rb') as f:
 | 
						|
        un, pw = f.read().strip().split(':')
 | 
						|
    return {
 | 
						|
        'username':un, 'password':pw
 | 
						|
    }
 | 
						|
 | 
						|
def get_sourceforge_data():
 | 
						|
    return {'username':'kovidgoyal', 'project':'calibre'}
 | 
						|
 | 
						|
def send_data(loc):
 | 
						|
    subprocess.check_call(['rsync', '--inplace', '--delete', '-r', '-z', '-h', '--progress', '-e', 'ssh -x',
 | 
						|
        loc+'/', '%s@%s:%s'%(STAGING_USER, STAGING_HOST, STAGING_DIR)])
 | 
						|
 | 
						|
def gc_cmdline(ver, gdata):
 | 
						|
    return [__appname__, ver, 'fmap', 'googlecode',
 | 
						|
                gdata['project'], gdata['username'], gdata['password'],
 | 
						|
                gdata['gc_password'], '--path-map-server',
 | 
						|
                gdata['path_map_server'], '--path-map-location',
 | 
						|
                gdata['path_map_location']]
 | 
						|
 | 
						|
def gh_cmdline(ver, data):
 | 
						|
    return [__appname__, ver, 'fmap', 'github', __appname__, data['username'], data['password']]
 | 
						|
 | 
						|
def sf_cmdline(ver, sdata):
 | 
						|
    return [__appname__, ver, 'fmap', 'sourceforge', sdata['project'],
 | 
						|
            sdata['username']]
 | 
						|
 | 
						|
def calibre_cmdline(ver):
 | 
						|
    return [__appname__, ver, 'fmap', 'calibre']
 | 
						|
 | 
						|
def dbs_cmdline(ver):
 | 
						|
    return [__appname__, ver, 'fmap', 'dbs']
 | 
						|
 | 
						|
def run_remote_upload(args):
 | 
						|
    print 'Running remotely:', ' '.join(args)
 | 
						|
    subprocess.check_call(['ssh', '-x', '%s@%s'%(STAGING_USER, STAGING_HOST),
 | 
						|
        'cd', STAGING_DIR, '&&', 'python', 'hosting.py']+args)
 | 
						|
 | 
						|
# }}}
 | 
						|
 | 
						|
class UploadInstallers(Command):  # {{{
 | 
						|
 | 
						|
    def add_options(self, parser):
 | 
						|
        parser.add_option('--replace', default=False, action='store_true', help='Replace existing installers')
 | 
						|
 | 
						|
    def run(self, opts):
 | 
						|
        all_possible = set(installers())
 | 
						|
        available = set(glob.glob('dist/*'))
 | 
						|
        files = {x:installer_description(x) for x in
 | 
						|
                all_possible.intersection(available)}
 | 
						|
        for x in files:
 | 
						|
            os.chmod(x, stat.S_IRUSR|stat.S_IWUSR|stat.S_IRGRP|stat.S_IROTH)
 | 
						|
        sizes = {os.path.basename(x):os.path.getsize(x) for x in files}
 | 
						|
        self.record_sizes(sizes)
 | 
						|
        tdir = mkdtemp()
 | 
						|
        backup = os.path.join('/mnt/external/calibre/%s' % __version__)
 | 
						|
        if not os.path.exists(backup):
 | 
						|
            os.mkdir(backup)
 | 
						|
        try:
 | 
						|
            self.upload_to_staging(tdir, backup, files)
 | 
						|
            self.upload_to_calibre()
 | 
						|
            if opts.replace:
 | 
						|
                upload_signatures()
 | 
						|
            self.upload_to_sourceforge()
 | 
						|
            self.upload_to_dbs()
 | 
						|
            self.upload_to_github(opts.replace)
 | 
						|
            # self.upload_to_google(opts.replace)
 | 
						|
        finally:
 | 
						|
            shutil.rmtree(tdir, ignore_errors=True)
 | 
						|
 | 
						|
    def record_sizes(self, sizes):
 | 
						|
        print ('\nRecording dist sizes')
 | 
						|
        args = ['%s:%s:%s' % (__version__, fname, size) for fname, size in sizes.iteritems()]
 | 
						|
        check_call(['ssh', 'divok', 'dist_sizes'] + args)
 | 
						|
 | 
						|
    def upload_to_staging(self, tdir, backup, files):
 | 
						|
        os.mkdir(tdir+'/dist')
 | 
						|
        hosting = os.path.join(os.path.dirname(os.path.abspath(__file__)),
 | 
						|
            'hosting.py')
 | 
						|
        shutil.copyfile(hosting, os.path.join(tdir, 'hosting.py'))
 | 
						|
 | 
						|
        for f in files:
 | 
						|
            for x in (tdir+'/dist', backup):
 | 
						|
                dest = os.path.join(x, os.path.basename(f))
 | 
						|
                shutil.copy2(f, x)
 | 
						|
                os.chmod(dest, stat.S_IREAD|stat.S_IWRITE|stat.S_IRGRP|stat.S_IROTH)
 | 
						|
 | 
						|
        with open(os.path.join(tdir, 'fmap'), 'wb') as fo:
 | 
						|
            for f, desc in files.iteritems():
 | 
						|
                fo.write('%s: %s\n'%(f, desc))
 | 
						|
 | 
						|
        while True:
 | 
						|
            try:
 | 
						|
                send_data(tdir)
 | 
						|
            except:
 | 
						|
                print('\nUpload to staging failed, retrying in a minute')
 | 
						|
                time.sleep(60)
 | 
						|
            else:
 | 
						|
                break
 | 
						|
 | 
						|
    def upload_to_google(self, replace):
 | 
						|
        gdata = get_google_data()
 | 
						|
        args = gc_cmdline(__version__, gdata)
 | 
						|
        if replace:
 | 
						|
            args = ['--replace'] + args
 | 
						|
        run_remote_upload(args)
 | 
						|
 | 
						|
    def upload_to_github(self, replace):
 | 
						|
        data = get_github_data()
 | 
						|
        args = gh_cmdline(__version__, data)
 | 
						|
        if replace:
 | 
						|
            args = ['--replace'] + args
 | 
						|
        run_remote_upload(args)
 | 
						|
 | 
						|
    def upload_to_sourceforge(self):
 | 
						|
        sdata = get_sourceforge_data()
 | 
						|
        args = sf_cmdline(__version__, sdata)
 | 
						|
        run_remote_upload(args)
 | 
						|
 | 
						|
    def upload_to_calibre(self):
 | 
						|
        run_remote_upload(calibre_cmdline(__version__))
 | 
						|
 | 
						|
    def upload_to_dbs(self):
 | 
						|
        run_remote_upload(dbs_cmdline(__version__))
 | 
						|
# }}}
 | 
						|
 | 
						|
class UploadUserManual(Command):  # {{{
 | 
						|
    description = 'Build and upload the User Manual'
 | 
						|
    sub_commands = ['manual']
 | 
						|
 | 
						|
    def build_plugin_example(self, path):
 | 
						|
        from calibre import CurrentDir
 | 
						|
        with NamedTemporaryFile(suffix='.zip') as f:
 | 
						|
            os.fchmod(f.fileno(),
 | 
						|
                stat.S_IRUSR|stat.S_IRGRP|stat.S_IROTH|stat.S_IWRITE)
 | 
						|
            with CurrentDir(path):
 | 
						|
                with ZipFile(f, 'w') as zf:
 | 
						|
                    for x in os.listdir('.'):
 | 
						|
                        if x.endswith('.swp'):
 | 
						|
                            continue
 | 
						|
                        zf.write(x)
 | 
						|
                        if os.path.isdir(x):
 | 
						|
                            for y in os.listdir(x):
 | 
						|
                                zf.write(os.path.join(x, y))
 | 
						|
            bname = self.b(path) + '_plugin.zip'
 | 
						|
            dest = '%s/%s'%(DOWNLOADS, bname)
 | 
						|
            subprocess.check_call(['scp', f.name, 'divok:'+dest])
 | 
						|
 | 
						|
    def run(self, opts):
 | 
						|
        path = self.j(self.SRC, '..', 'manual', 'plugin_examples')
 | 
						|
        for x in glob.glob(self.j(path, '*')):
 | 
						|
            self.build_plugin_example(x)
 | 
						|
 | 
						|
        srcdir = self.j(gettempdir(), 'user-manual-build', 'en', 'html') + '/'
 | 
						|
        for host in ('download', 'files'):
 | 
						|
            check_call(' '.join(['rsync', '-zrl', '--progress',
 | 
						|
                srcdir, '%s:/srv/manual/' % host]), shell=True)
 | 
						|
# }}}
 | 
						|
 | 
						|
class UploadDemo(Command):  # {{{
 | 
						|
 | 
						|
    description = 'Rebuild and upload various demos'
 | 
						|
 | 
						|
    def run(self, opts):
 | 
						|
        check_call(
 | 
						|
           '''ebook-convert %s/demo.html /tmp/html2lrf.lrf '''
 | 
						|
           '''--title='Demonstration of html2lrf' --authors='Kovid Goyal' '''
 | 
						|
           '''--header '''
 | 
						|
           '''--serif-family "/usr/share/fonts/corefonts, Times New Roman" '''
 | 
						|
           '''--mono-family  "/usr/share/fonts/corefonts, Andale Mono" '''
 | 
						|
           ''''''%self.j(self.SRC, HTML2LRF), shell=True)
 | 
						|
 | 
						|
        lrf = self.j(self.SRC, 'calibre', 'ebooks', 'lrf', 'html', 'demo')
 | 
						|
        check_call(
 | 
						|
            'cd %s && zip -j /tmp/html-demo.zip * /tmp/html2lrf.lrf' % lrf, shell=True)
 | 
						|
 | 
						|
        check_call('scp /tmp/html-demo.zip divok:%s/'%(DOWNLOADS,), shell=True)
 | 
						|
# }}}
 | 
						|
 | 
						|
class UploadToServer(Command):  # {{{
 | 
						|
 | 
						|
    description = 'Upload miscellaneous data to calibre server'
 | 
						|
 | 
						|
    def run(self, opts):
 | 
						|
        upload_signatures()
 | 
						|
        check_call('gpg --armor --detach-sign dist/calibre-*.tar.xz',
 | 
						|
                shell=True)
 | 
						|
        check_call('scp dist/calibre-*.tar.xz.asc divok:%s/signatures/'%DOWNLOADS,
 | 
						|
                shell=True)
 | 
						|
        check_call('ssh divok /usr/local/bin/update-calibre',
 | 
						|
                   shell=True)
 | 
						|
        check_call('''ssh divok echo %s \\> %s/latest_version'''
 | 
						|
                   %(__version__, DOWNLOADS), shell=True)
 | 
						|
        check_call('ssh divok /etc/init.d/apache2 graceful',
 | 
						|
                   shell=True)
 | 
						|
# }}}
 | 
						|
 | 
						|
# Testing {{{
 | 
						|
 | 
						|
def write_files(fmap):
 | 
						|
    for f in fmap:
 | 
						|
        with open(f, 'wb') as f:
 | 
						|
            f.write(os.urandom(100))
 | 
						|
            f.write(b'a'*1000000)
 | 
						|
    with open('fmap', 'wb') as fo:
 | 
						|
        for f, desc in fmap.iteritems():
 | 
						|
            fo.write('%s: %s\n'%(f, desc))
 | 
						|
 | 
						|
def setup_installers():
 | 
						|
    ver = '0.0.1'
 | 
						|
    files = {x.replace(__version__, ver):installer_description(x) for x in installers()}
 | 
						|
    tdir = mkdtemp()
 | 
						|
    os.chdir(tdir)
 | 
						|
    return tdir, files, ver
 | 
						|
 | 
						|
def test_google_uploader():
 | 
						|
    gdata = get_google_data()
 | 
						|
    gdata['project'] = 'calibre-hosting-uploader'
 | 
						|
    gdata['path_map_location'] += '-test'
 | 
						|
    hosting = os.path.join(os.path.dirname(os.path.abspath(__file__)),
 | 
						|
        'hosting.py')
 | 
						|
 | 
						|
    tdir, files, ver = setup_installers()
 | 
						|
    try:
 | 
						|
        os.mkdir('dist')
 | 
						|
        write_files(files)
 | 
						|
        shutil.copyfile(hosting, 'hosting.py')
 | 
						|
        send_data(tdir)
 | 
						|
        args = gc_cmdline(ver, gdata)
 | 
						|
 | 
						|
        print ('Doing initial upload')
 | 
						|
        run_remote_upload(args)
 | 
						|
        raw_input('Press Enter to proceed:')
 | 
						|
 | 
						|
        print ('\nDoing re-upload')
 | 
						|
        run_remote_upload(['--replace']+args)
 | 
						|
        raw_input('Press Enter to proceed:')
 | 
						|
 | 
						|
        nv = ver + '.1'
 | 
						|
        files = {x.replace(__version__, nv):installer_description(x) for x in installers()}
 | 
						|
        write_files(files)
 | 
						|
        send_data(tdir)
 | 
						|
        args[1] = nv
 | 
						|
        print ('\nDoing update upload')
 | 
						|
        run_remote_upload(args)
 | 
						|
        print ('\nDont forget to delete any remaining files in the %s project'%
 | 
						|
                gdata['project'])
 | 
						|
 | 
						|
    finally:
 | 
						|
        shutil.rmtree(tdir)
 | 
						|
# }}}
 | 
						|
 | 
						|
if __name__ == '__main__':
 | 
						|
    test_google_uploader()
 |