seafile-containerized/seafile-server/scripts/upgrade.py

144 lines
4.8 KiB
Python
Raw Normal View History

2021-01-17 23:45:50 +00:00
#!/usr/bin/env python3
#coding: UTF-8
"""
This script is used to run proper upgrade scripts automatically.
"""
import json
import re
import glob
import logging
2021-01-17 23:45:50 +00:00
import os
from os.path import abspath, basename, exists, dirname, join, isdir, islink
2021-01-17 23:45:50 +00:00
import sys
import time
from utils import (
call, get_install_dir, get_script, get_command_output, replace_file_pattern,
read_version_stamp, wait_for_mysql, update_version_stamp, loginfo
)
installdir = get_install_dir()
topdir = dirname(installdir)
logger = logging.getLogger(__name__)
2021-01-17 23:45:50 +00:00
def collect_upgrade_scripts(from_version, to_version):
"""
Give the current installed version, calculate which upgrade scripts we need
to run to upgrade it to the latest verison.
For example, given current version 5.0.1 and target version 6.1.0, and these
upgrade scripts:
upgrade_4.4_5.0.sh
upgrade_5.0_5.1.sh
upgrade_5.1_6.0.sh
upgrade_6.0_6.1.sh
We need to run upgrade_5.0_5.1.sh, upgrade_5.1_6.0.sh, and upgrade_6.0_6.1.sh.
"""
from_major_ver = '.'.join(from_version.split('.')[:2])
to_major_ver = '.'.join(to_version.split('.')[:2])
def version_str_to_float(filename):
v = filename.split('_')[1]
try:
return float(v)
except:
return str(v)
""" String '8.0' < '9.0' is True, but '9.0' < '10.0' is False in Python3.
So we need to convert the version string to float to compare.
fn: script filepath, eg: /opt/seafile/seafile-server-latest/upgrade/upgrade_10.0_11.0.sh
va: from_major_version in script filename, eg: 10.0
vb: to_major_version in script filename, eg: 11.0
"""
2021-01-17 23:45:50 +00:00
scripts = []
for fn in sorted(glob.glob(join(installdir, 'upgrade', 'upgrade_*_*.sh')), key=version_str_to_float):
2021-01-17 23:45:50 +00:00
va, vb = parse_upgrade_script_version(fn)
try:
va = float(va)
vb = float(vb)
from_major_ver = float(from_major_ver)
to_major_ver = float(to_major_ver)
except:
va = str(va)
vb = str(vb)
from_major_ver = str(from_major_ver)
to_major_ver = str(to_major_ver)
2021-01-17 23:45:50 +00:00
if va >= from_major_ver and vb <= to_major_ver:
scripts.append(fn)
return scripts
def parse_upgrade_script_version(script):
script = basename(script)
m = re.match(r'upgrade_([0-9+.]+)_([0-9+.]+).sh', basename(script))
return m.groups()
def run_script_and_update_version_stamp(script, new_version):
logging.info('Running script %s', script)
replace_file_pattern(script, 'read dummy', '')
call(script)
update_version_stamp(new_version)
def is_minor_upgrade(v1, v2):
get_major_version = lambda x: x.split('.')[:2]
return v1 != v2 and get_major_version(v1) == get_major_version(v2)
def fix_media_symlinks():
"""
If the container was recreated and it's not a minor/major upgrade,
we need to fix the media/avatars and media/custom symlink.
"""
media_dir = join(installdir, 'seahub/media')
avatars_dir = join(media_dir, 'avatars')
custom_dir = join(media_dir, 'custom')
dst_media_dir = ('/shared/seafile/seahub-data')
dst_avatars_dir = join(dst_media_dir, 'avatars')
dst_custom_dir = join(dst_media_dir, 'custom')
if not islink(avatars_dir):
logger.info('The container was recreated, start fix the media symlinks')
call('mv -n %s/* %s' % (avatars_dir, dst_avatars_dir))
call('rm -rf %s' % avatars_dir)
call('ln -sf %s %s' % (dst_avatars_dir, avatars_dir))
call('rm -rf %s' % custom_dir)
call('ln -sf %s %s' % (dst_custom_dir, custom_dir))
logger.info('Done')
def run_minor_upgrade(current_version):
minor_upgrade_script = join(installdir, 'upgrade', 'minor-upgrade.sh')
run_script_and_update_version_stamp(minor_upgrade_script, current_version)
2021-01-17 23:45:50 +00:00
def check_upgrade():
last_version = read_version_stamp()
current_version = os.environ['SEAFILE_VERSION']
2021-01-17 23:45:50 +00:00
if last_version == current_version:
fix_media_symlinks()
return
elif is_minor_upgrade(last_version, current_version):
run_minor_upgrade(current_version)
2021-01-17 23:45:50 +00:00
return
# Now we do the major upgrade
2021-01-17 23:45:50 +00:00
scripts_to_run = collect_upgrade_scripts(from_version=last_version, to_version=current_version)
for script in scripts_to_run:
loginfo('Running scripts {}'.format(script))
# Here we use a trick: use a version stamp like 6.1.0 to prevent running
# all upgrade scripts before 6.1 again (because 6.1 < 6.1.0 in python)
new_version = parse_upgrade_script_version(script)[1] + '.0'
run_script_and_update_version_stamp(script, new_version)
2021-01-17 23:45:50 +00:00
update_version_stamp(current_version)
def main():
wait_for_mysql()
os.chdir(installdir)
check_upgrade()
if __name__ == '__main__':
main()