Files
entropy/server/eit/commands/push.py
2011-10-17 22:28:30 +02:00

262 lines
9.0 KiB
Python

# -*- coding: utf-8 -*-
"""
@author: Fabio Erculiani <lxnay@sabayon.org>
@contact: lxnay@sabayon.org
@copyright: Fabio Erculiani
@license: GPL-2
B{Entropy Infrastructure Toolkit}.
"""
import sys
import os
import argparse
import tempfile
from entropy.const import etpConst
from entropy.i18n import _
from entropy.output import darkgreen, teal, red, darkred, brown, blue, \
bold, purple
from entropy.transceivers import EntropyTransceiver
from entropy.server.interfaces import ServerSystemSettingsPlugin
from entropy.server.interfaces.rss import ServerRssMetadata
from eit.commands.descriptor import EitCommandDescriptor
from eit.commands.command import EitCommand
class EitPush(EitCommand):
"""
Main Eit reset command.
"""
NAME = "push"
ALIASES = ["pull","sync"]
DEFAULT_REPO_COMMIT_MSG = """
# This is Entropy Server repository commit message handler.
# Please friggin' enter the commit message for your changes. Lines starting
# with '#' will be ignored. To avoid encoding issue, write stuff in plain ASCII.
"""
def __init__(self, args):
EitCommand.__init__(self, args)
self._ask = True
self._pretend = False
self._all = False
self._repositories = []
self._cleanup_only = False
self._as_repository_id = None
def _get_parser(self):
descriptor = EitCommandDescriptor.obtain_descriptor(
EitPush.NAME)
parser = argparse.ArgumentParser(
description=descriptor.get_description(),
formatter_class=argparse.RawDescriptionHelpFormatter,
prog="%s %s" % (sys.argv[0], EitPush.NAME))
parser.add_argument("repo", nargs='?', default=None,
metavar="<repo>", help=_("repository"))
parser.add_argument("--quick", action="store_true",
default=False,
help=_("no stupid questions"))
group = parser.add_mutually_exclusive_group()
group.add_argument("--all", action="store_true",
default=False,
help=_("push all the repositories"))
group.add_argument("--as", metavar="<repo>", default=None,
help=_("push as fake repository"), dest="asrepo")
return parser
def parse(self):
parser = self._get_parser()
try:
nsargs = parser.parse_args(self._args)
except IOError as err:
sys.stderr.write("%s\n" % (err,))
return parser.print_help, []
self._ask = not nsargs.quick
self._all = nsargs.all
if nsargs.repo is not None:
self._repositories.append(nsargs.repo)
self._as_repository_id = nsargs.asrepo
return self._call_locked, [self._push, nsargs.repo]
def _push(self, entropy_server):
"""
Main Eit push code.
"""
if not self._repositories and (not self._all):
# pick default if none specified
self._repositories.append(entropy_server.repository())
if not self._repositories and self._all:
self._repositories.extend(entropy_server.repositories())
for repository_id in self._repositories:
# avoid __default__
if repository_id == etpConst['clientserverrepoid']:
continue
rc = self._push_repo(entropy_server, repository_id)
if rc != 0:
return rc
return 0
def _push_repo(self, entropy_server, repository_id):
"""
Push (actually sync) the repository.
"""
done = True
if not self._cleanup_only:
rc = self.__push_repo(entropy_server, repository_id)
if rc != 0:
done = False
if done:
done = entropy_server.Mirrors.tidy_mirrors(repository_id,
ask = self._ask, pretend = self._pretend)
if done:
return 0
return 1
def _commit_message(self, entropy_server, successfull_mirrors):
"""
Ask user to enter the commit message for data being pushed.
Store inside rss metadata object.
"""
tmp_fd, tmp_commit_path = tempfile.mkstemp(
prefix="eit._push", suffix=".COMMIT_MSG")
with os.fdopen(tmp_fd, "w") as tmp_f:
tmp_f.write(EitPush.DEFAULT_REPO_COMMIT_MSG)
if successfull_mirrors:
tmp_f.write("# Changes to be committed:\n")
for sf_mirror in sorted(successfull_mirrors):
tmp_f.write("#\t updated: %s\n" % (sf_mirror,))
# spawn editor
cm_msg_rc = entropy_server.edit_file(tmp_commit_path)
commit_msg = None
if not cm_msg_rc:
# wtf?, fallback to old way
def fake_callback(*args, **kwargs):
return True
input_params = [
('message', _("Commit message"), fake_callback, False)]
commit_data = entropy_server.input_box(
_("Enter the commit message"),
input_params, cancel_button = True)
if commit_data:
commit_msg = commit_data['message']
else:
commit_msg = ''
with open(tmp_commit_path, "r") as tmp_f:
for line in tmp_f.readlines():
if line.strip().startswith("#"):
continue
commit_msg += line
entropy_server.output(commit_msg)
os.remove(tmp_commit_path)
return commit_msg
def __print_repository_status(self, entropy_server, repository_id):
remote_db_status = entropy_server.Mirrors.remote_repository_status(
repository_id)
entropy_server.output(
"%s:" % (brown(_("Entropy Repository Status")),),
importance=1,
header=darkgreen(" * ")
)
for url, revision in remote_db_status.items():
host = EntropyTransceiver.get_uri_name(url)
entropy_server.output(
"%s: %s" % (darkgreen(_("Host")), bold(host)),
header=" ")
entropy_server.output(
"%s: %s" % (purple(_("Remote")), blue(str(revision))),
header=" ")
local_revision = entropy_server.local_repository_revision(
repository_id)
entropy_server.output(
"%s: %s" % (brown(_("Local")), teal(str(local_revision))),
header=" ")
def __sync_remote_database(self, entropy_server, repository_id):
self.__print_repository_status(entropy_server, repository_id)
# do the actual sync
sts = entropy_server.Mirrors.sync_repository(repository_id)
self.__print_repository_status(entropy_server, repository_id)
return sts
def __push_repo(self, entropy_server, repository_id):
sys_settings_plugin_id = \
etpConst['system_settings_plugins_ids']['server_plugin']
srv_data = self._settings()[sys_settings_plugin_id]['server']
rss_enabled = srv_data['rss']['enabled']
mirrors_tainted, mirrors_errors, successfull_mirrors, \
broken_mirrors, check_data = \
entropy_server.Mirrors.sync_packages(
repository_id, ask = self._ask,
pretend = self._pretend)
if mirrors_errors and not successfull_mirrors:
entropy_server.output(red(_("Aborting !")),
importance=1, level="error", header=darkred(" !!! "))
return 1
if not successfull_mirrors:
return 0
if mirrors_tainted and (self._as_repository_id is None):
commit_msg = None
if self._ask and rss_enabled:
commit_msg = self._commit_message(entropy_server,
successfull_mirrors)
elif rss_enabled:
commit_msg = "Automatic update"
if commit_msg is None:
commit_msg = "no commit message"
ServerRssMetadata()['commitmessage'] = commit_msg
if self._as_repository_id is not None:
# change repository push location
ServerSystemSettingsPlugin.set_override_remote_repository(
self._settings(), repository_id, self._as_repository_id)
sts = self.__sync_remote_database(entropy_server, repository_id)
if sts == 0:
# do not touch locking
entropy_server.Mirrors.lock_mirrors(repository_id, False,
unlock_locally = (self._as_repository_id is None))
if (sts == 0) and self._ask:
q_rc = entropy_server.ask_question(
_("Should I cleanup old packages on mirrors ?"))
if q_rc == _("No"):
return 0
elif sts != 0:
entropy_server.output(red(_("Aborting !")),
importance=1, level="error", header=darkred(" !!! "))
return sts
return 0
EitCommandDescriptor.register(
EitCommandDescriptor(
EitPush,
EitPush.NAME,
_('push (or pull) repository packages and metadata'))
)