Compare commits

...

2 Commits

3 changed files with 81 additions and 43 deletions

View File

@@ -342,7 +342,7 @@ def do_transfer_offer():
host, user, port_int = resume_host, resume_user or "root", int(resume_port) host, user, port_int = resume_host, resume_user or "root", int(resume_port)
else: else:
# Сбрасываем, чтобы запросить заново # Сбрасываем, чтобы запросить заново
state.set_stage("SOURCE_STOP", target_host=None, target_user=None, target_port=None) state.set_stage("SOURCE_STOP", target_host=None, target_user=None, target_port=None, ssh_key=None)
host = user = None host = user = None
else: else:
host = user = None host = user = None
@@ -378,8 +378,19 @@ def do_transfer_offer():
if key_path: if key_path:
state.set_stage("TRANSFER", target_host=host, target_user=user, target_port=port_int, ssh_key=key_path) state.set_stage("TRANSFER", target_host=host, target_user=user, target_port=port_int, ssh_key=key_path)
from transfer.transfer import do_transfer
do_transfer()
else: else:
state.set_stage("TRANSFER", target_host=host, target_user=user, target_port=port_int) # Нет SSH-ключа — спрашиваем пароль и пробуем sshpass
password = prompt(f"Введите пароль для {user}@{host} (или Enter для отмены)")
from transfer.transfer import do_transfer if not password:
do_transfer() state.set_error("ssh_key_setup", "", "Пароль не введён", suggestion="Запустите docker-migrate --resume или настройте SSH-ключ")
raise RuntimeError("Пароль не введён. Установите SSH-ключ или введите пароль.")
# Проверяем/устанавливаем sshpass
from transfer.ssh import ensure_sshpass
if not ensure_sshpass():
state.set_error("ssh_key_setup", "", "sshpass не найден и не удалось установить", suggestion="Установите sshpass: apt-get install -y sshpass")
raise RuntimeError("sshpass не найден. Установите: apt-get install -y sshpass")
state.set_stage("TRANSFER", target_host=host, target_user=user, target_port=port_int, ssh_key=None, ssh_password=password)
from transfer.transfer import do_transfer
do_transfer()

View File

@@ -147,6 +147,21 @@ def manual_copy_instructions(host, user, port, pubkey_path):
print() print()
def ensure_sshpass():
"""Проверяет наличие sshpass, при необходимости ставит через apt."""
if exists("sshpass"):
return True
info("sshpass не найден. Устанавливаем ...")
r1 = run("apt-get update", check=False, timeout=60)
if r1.returncode != 0:
warn(f"apt-get update не удался: {r1.stderr.strip()[:120]}")
r = run("apt-get install -y sshpass", check=False, timeout=120)
if r.returncode != 0:
warn(f"Не удалось установить sshpass: {r.stderr.strip()[:120]}")
return False
return exists("sshpass")
def pick_or_setup_ssh_key(host, user, port): def pick_or_setup_ssh_key(host, user, port):
""" """
Главная функция SSH-подготовки. Главная функция SSH-подготовки.

View File

@@ -19,6 +19,8 @@ def do_transfer():
host = st.get("target_host") host = st.get("target_host")
user = st.get("target_user") user = st.get("target_user")
port = st.get("target_port", 22) port = st.get("target_port", 22)
key_path = st.get("ssh_key")
password = st.get("ssh_password")
if not host or host.lower() in ("none", "", "localhost"): if not host or host.lower() in ("none", "", "localhost"):
warn("Target host не указан. Перенос невозможен.") warn("Target host не указан. Перенос невозможен.")
@@ -36,14 +38,31 @@ def do_transfer():
size_mb = size / 1024 / 1024 size_mb = size / 1024 / 1024
info(f"Размер архива: {size_mb:.2f} MB") info(f"Размер архива: {size_mb:.2f} MB")
# Определяем путь назначения # Путь назначения
remote_dir = "/tmp/docker-migrate-incoming" remote_dir = "/tmp/docker-migrate-incoming"
remote_path = os.path.join(remote_dir, os.path.basename(archive_path)) remote_path = os.path.join(remote_dir, os.path.basename(archive_path))
# Копируем bootstrap (migrate + core/) чтобы target мог запуститься # Создаём remote_dir через ssh перед scp/rsync
# Найдём корень проекта ssh_base = f"ssh -p {port} -o ConnectTimeout=10 -o StrictHostKeyChecking=accept-new"
project_root = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) if key_path and os.path.isfile(key_path):
tar_script = os.path.join(project_root, "tar_project.sh") ssh_base += f" -i '{key_path}'"
info(f"Используем SSH-ключ: {key_path}")
elif password:
if not exists("sshpass"):
raise RuntimeError("sshpass не найден, но нужен для пароля. Установите: apt-get install -y sshpass")
ssh_base = f"sshpass -p '{password}' {ssh_base}"
info("Используем пароль через sshpass")
else:
ssh_base += " -o BatchMode=yes"
info("Пробуем ключевую аутентификацию (BatchMode)")
# Проверим что target отвечает (mkdir)
info("Проверяем доступ к target ...")
r0 = run(f"{ssh_base} {user}@{host} 'mkdir -p {remote_dir}'", check=False, timeout=30)
if r0.returncode != 0:
cerror(f"SSH не удался: {r0.stderr.strip()[:200]}")
state.set_error("TRANSFER", r0.stdout, r0.stderr, suggestion="Проверьте пароль/ключ и доступ SSH. После исправления: docker-migrate --resume")
raise RuntimeError(f"SSH-доступ не работает. Код: {r0.returncode}")
# Выбираем метод # Выбираем метод
if size_mb < 50 and exists("scp"): if size_mb < 50 and exists("scp"):
@@ -60,28 +79,29 @@ def do_transfer():
info(f"Выбран метод переноса: {method}") info(f"Выбран метод переноса: {method}")
key_path = st.get("ssh_key")
# Предварительная проверка уже сделана в pick_or_setup_ssh_key, повторно не делаем
# Базовые SSH-опции (ssh/rsh/rsync)
ssh_opts = f"-p {port}"
if key_path and os.path.isfile(key_path):
ssh_opts += f" -i '{key_path}'"
info(f"Используем SSH-ключ: {key_path}")
if method == "scp": if method == "scp":
# scp использует заглавную -P для порта (строчная -p = preserve timestamps) # scp: заглавная -P для порта
# Добавляем SSH-опции чтобы избежать зависания и silent fail scp_opts = f"-P {port} -o ConnectTimeout=10 -o StrictHostKeyChecking=accept-new"
scp_opts = f"-P {port} -o ConnectTimeout=10 -o StrictHostKeyChecking=accept-new -o BatchMode=yes"
if key_path and os.path.isfile(key_path): if key_path and os.path.isfile(key_path):
scp_opts += f" -i '{key_path}'" scp_opts += f" -i '{key_path}'"
info("Копируем архив через scp ...") info("Копируем архив через scp ...")
r = run(f"scp {scp_opts} {archive_path} {user}@{host}:{remote_path}", check=False, timeout=60) r = run(f"scp {scp_opts} {archive_path} {user}@{host}:{remote_path}", check=False, timeout=60)
elif password:
# sshpass оборачивает всю команду scp
info("Копируем архив через scp + sshpass ...")
r = run(f"sshpass -p '{password}' scp {scp_opts} {archive_path} {user}@{host}:{remote_path}", check=False, timeout=60)
else:
scp_opts += " -o BatchMode=yes"
info("Копируем архив через scp ...")
r = run(f"scp {scp_opts} {archive_path} {user}@{host}:{remote_path}", check=False, timeout=60)
else: else:
# rsync через ssh_base
info("Копируем архив через rsync ...") info("Копируем архив через rsync ...")
r = run(f"rsync -avz --progress -e 'ssh {ssh_opts}' {archive_path} {user}@{host}:{remote_path}", check=False, timeout=60) rsync_ssh = ssh_base.replace("ssh ", "") # убираем 'ssh ' для rsync -e
r = run(f"rsync -avz --progress -e 'ssh {rsync_ssh}' {archive_path} {user}@{host}:{remote_path}", check=False, timeout=60)
if r.returncode != 0: if r.returncode != 0:
cerror(f"Передача не удалась: {r.stderr.strip()[:200]}")
state.set_error( state.set_error(
step="TRANSFER", step="TRANSFER",
stdout=r.stdout, stdout=r.stdout,
@@ -92,33 +112,25 @@ def do_transfer():
success(f"Архив передан на {host}:{remote_path}") success(f"Архив передан на {host}:{remote_path}")
# Распаковка на target во временную директорию # Распаковка на target
info("Распаковываем архив на target ...") info("Распаковываем архив на target ...")
r2 = run(f"ssh {ssh_opts} {user}@{host} 'mkdir -p {remote_dir} && tar xzf {remote_path} -C {remote_dir}'", check=False) r2 = run(f"{ssh_base} {user}@{host} 'tar xzf {remote_path} -C {remote_dir}'", check=False, timeout=60)
if r2.returncode != 0: if r2.returncode != 0:
warn(f"Не удалось распаковать архив на target: {r2.stderr}") warn(f"Не удалось распаковать: {r2.stderr.strip()[:200]}")
state.set_error( state.set_error("TRANSFER_UNPACK", r2.stdout, r2.stderr, suggestion="Проверьте tar на target")
step="TRANSFER_UNPACK", raise RuntimeError("Распаковка не удалась")
stdout=r2.stdout,
stderr=r2.stderr,
suggestion=f"Проверьте доступ SSH и tar на {host}. После исправления: docker-migrate --resume"
)
raise RuntimeError(f"Распаковка на target не удалась: {r2.stderr}")
# Сохраняем remote_dir в state для target # Сохраняем remote_dir в state для target
state.set_stage("TRANSFER", target_remote_dir=remote_dir) state.set_stage("TRANSFER", target_remote_dir=remote_dir)
# Предлагаем сразу запустить target-режим удалённо # Предлагаем запустить target-режим удалённо
if confirm("Сразу запустить восстановление на новом сервере (remote target mode)", default="y"): if confirm("Сразу запустить восстановление на новом сервере", default="y"):
info("Запускаем target-mode удалённо ...") info("Запускаем target-mode удалённо ...")
# Передаём скрипт на target
project_local = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) project_local = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
run(f"scp {ssh_opts} -r {project_local} {user}@{host}:/opt/docker-migrate-tool", check=False) run(f"{ssh_base} {user}@{host} 'mkdir -p /opt/docker-migrate-tool && tar czf - -C {project_local} . | tar xzf - -C /opt/docker-migrate-tool'", check=False, timeout=60)
# Запуск target как remote python run(f"{ssh_base} {user}@{host} 'cd /opt/docker-migrate-tool && python3 core/main.py --mode=target --remote-dir={remote_dir}'", check=False, timeout=300)
run(f"ssh {ssh_opts} {user}@{host} 'cd /opt/docker-migrate-tool && python3 core/main.py --mode=target --remote-dir={remote_dir}'", check=False)
else: else:
success(f"Архив передан. Запустите на target: python3 core/main.py --mode=target --remote-dir={remote_dir}") success(f"Архив передан. Запустите на target: python3 core/main.py --mode=target --remote-dir={remote_dir}")
# Transfer выполнен — отмечаем выполненным, чтобы resume не повторял
state.mark_completed("TRANSFER") state.mark_completed("TRANSFER")
state.set_stage("DONE") state.set_stage("DONE")