1
1
# Copyright (C) 2006 by Szilveszter Farkas (Phanatic) <szilveszter.farkas@gmail.com>
2
# Copyright (C) 2007 by Jelmer Vernooij <jelmer@samba.org>
3
4
# This program is free software; you can redistribute it and/or modify
4
5
# it under the terms of the GNU General Public License as published by
5
6
# the Free Software Foundation; either version 2 of the License, or
6
7
# (at your option) any later version.
8
9
# This program is distributed in the hope that it will be useful,
9
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
12
# GNU General Public License for more details.
13
14
# You should have received a copy of the GNU General Public License
14
15
# along with this program; if not, write to the Free Software
15
16
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21
20
pygtk.require("2.0")
31
import olive.backend.commit as commit
32
import olive.backend.errors as errors
35
""" Display Push dialog and perform the needed actions. """
36
def __init__(self, gladefile, comm):
26
from errors import show_bzr_error
28
# FIXME: This needs to be public JRV 20070714
29
from bzrlib.builtins import _create_prefix
30
from bzrlib.config import LocationConfig
31
import bzrlib.errors as errors
33
from dialog import error_dialog, info_dialog, question_dialog
35
from history import UrlHistory
37
class PushDialog(gtk.Dialog):
38
""" New implementation of the Push dialog. """
39
def __init__(self, repository, revid, branch=None, parent=None):
37
40
""" Initialize the Push dialog. """
38
self.gladefile = gladefile
39
self.glade = gtk.glade.XML(self.gladefile, 'window_push')
43
self.window = self.glade.get_widget('window_push')
45
# Dictionary for signal_autoconnect
46
dic = { "on_button_push_push_clicked": self.push,
47
"on_button_push_cancel_clicked": self.close,
48
"on_radiobutton_push_stored_toggled": self.stored_toggled,
49
"on_radiobutton_push_specific_toggled": self.specific_toggled, }
51
# Connect the signals to the handlers
52
self.glade.signal_autoconnect(dic)
54
# Get some useful widgets
55
self.entry_location = self.glade.get_widget('entry_push_location')
56
self.check_remember = self.glade.get_widget('checkbutton_push_remember')
57
self.check_overwrite = self.glade.get_widget('checkbutton_push_overwrite')
58
self.check_create = self.glade.get_widget('checkbutton_push_create')
61
""" Display the Push dialog. """
63
self.width, self.height = self.window.get_size()
65
def stored_toggled(self, widget):
66
if widget.get_active():
67
self.entry_location.hide()
68
self.check_remember.hide()
69
self.check_overwrite.hide()
70
self.check_create.hide()
71
self.window.resize(self.width, self.height)
73
self.entry_location.show()
74
self.check_remember.show()
75
self.check_overwrite.show()
76
self.check_create.show()
78
def specific_toggled(self, widget):
79
if widget.get_active():
80
self.entry_location.show()
81
self.check_remember.show()
82
self.check_overwrite.show()
83
self.check_create.show()
85
self.entry_location.hide()
86
self.check_remember.hide()
87
self.check_overwrite.hide()
88
self.check_create.hide()
90
def push(self, widget):
91
from dialog import OliveDialog
92
dialog = OliveDialog(self.gladefile)
94
radio_stored = self.glade.get_widget('radiobutton_push_stored')
95
radio_specific = self.glade.get_widget('radiobutton_push_specific')
41
gtk.Dialog.__init__(self, title="Push - Olive",
44
buttons=(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL))
47
self.repository = repository
52
self._label_location = gtk.Label(_("Location:"))
53
self._combo = gtk.ComboBoxEntry()
54
self._button_push = gtk.Button(_("_Push"), use_underline=True)
55
self._hbox_location = gtk.HBox()
58
self._button_push.connect('clicked', self._on_push_clicked)
61
self._label_location.set_alignment(0, 0.5)
62
self._hbox_location.set_spacing(3)
63
self.vbox.set_spacing(3)
66
self._hbox_location.pack_start(self._label_location, False, False)
67
self._hbox_location.pack_start(self._combo, True, True)
68
self.vbox.pack_start(self._hbox_location)
69
self.action_area.pack_end(self._button_push)
74
# Build location history
75
self._history = UrlHistory(self.branch.get_config(), 'push_history')
78
def _build_history(self):
79
""" Build up the location history. """
80
self._combo_model = gtk.ListStore(str)
81
for item in self._history.get_entries():
82
self._combo_model.append([ item ])
83
self._combo.set_model(self._combo_model)
84
self._combo.set_text_column(0)
86
if self.branch is not None:
87
location = self.branch.get_push_location()
88
if location is not None:
89
self._combo.get_child().set_text(location)
92
def _on_push_clicked(self, widget):
93
""" Push button clicked handler. """
94
location = self._combo.get_child().get_text()
98
self.comm.set_busy(self.window)
99
if radio_stored.get_active():
101
revs = commit.push(self.comm.get_path())
102
except errors.NotBranchError:
103
dialog.error_dialog('Directory is not a branch.')
105
except errors.NoLocationKnown:
106
dialog.error_dialog('No location known.')
108
except errors.NonExistingParent, errmsg:
109
dialog.error_dialog('Parent directory doesn\'t exist: %s', errmsg)
111
except errors.DivergedBranchesError:
112
dialog.error_dialog('Branches have been diverged.')
116
elif radio_specific.get_active():
117
location = self.entry_location.get_text()
119
dialog.error_dialog('No location specified.')
123
revs = commit.push(self.comm.get_path(), location,
124
self.check_remember.get_active(),
125
self.check_overwrite.get_active(),
126
self.check_create.get_active())
127
except errors.NotBranchError:
128
dialog.error_dialog('Directory is not a branch.')
129
self.comm.set_busy(self.window, False)
131
except errors.NonExistingParent, errmsg:
132
dialog.error_dialog('Parent directory doesn\'t exist: %s', errmsg)
133
self.comm.set_busy(self.window, False)
135
except errors.DivergedBranchesError:
136
dialog.error_dialog('Branches have been diverged.')
137
self.comm.set_busy(self.window, False)
139
except errors.PathPrefixNotCreated:
140
dialog.error_dialog('Path prefix couldn\'t be created.')
141
self.comm.set_busy(self.window, False)
96
if self.branch is not None and self.branch.get_push_location() is None:
97
response = question_dialog(_('Set default push location'),
98
_('There is no default push location set.\nSet %r as default now?') % location)
99
if response == gtk.RESPONSE_OK:
100
self.branch.set_push_location(location)
103
revs = do_push(self.branch, location=location, overwrite=False)
104
except errors.DivergedBranches:
105
response = question_dialog(_('Branches have been diverged'),
106
_('You cannot push if branches have diverged.\nOverwrite?'))
107
if response == gtk.RESPONSE_YES:
108
revs = do_push(self.branch, location=location, overwrite=True)
110
self._history.add_entry(location)
111
info_dialog(_('Push successful'),
112
_("%d revision(s) pushed.") % revs)
114
self.response(gtk.RESPONSE_OK)
116
def do_push(br_from, location, overwrite):
117
""" Update a mirror of a branch.
119
:param br_from: the source branch
121
:param location: the location of the branch that you'd like to update
123
:param overwrite: overwrite target location if it diverged
125
:return: number of revisions pushed
127
from bzrlib.bzrdir import BzrDir
128
from bzrlib.transport import get_transport
130
transport = get_transport(location)
131
location_url = transport.base
136
dir_to = BzrDir.open(location_url)
137
br_to = dir_to.open_branch()
138
except errors.NotBranchError:
140
transport = transport.clone('..')
142
relurl = transport.relpath(location_url)
143
transport.mkdir(relurl)
144
except errors.NoSuchFile:
145
response = question_dialog(_('Non existing parent directory'),
146
_("The parent directory (%s)\ndoesn't exist. Create?") % location)
147
if response == gtk.RESPONSE_OK:
148
_create_prefix(transport)
151
dir_to = br_from.bzrdir.clone(location_url,
152
revision_id=br_from.last_revision())
153
br_to = dir_to.open_branch()
154
count = len(br_to.revision_history())
156
old_rh = br_to.revision_history()
158
tree_to = dir_to.open_workingtree()
159
except errors.NotLocalUrl:
160
# FIXME - what to do here? how should we warn the user?
161
count = br_to.pull(br_from, overwrite)
162
except errors.NoWorkingTree:
163
count = br_to.pull(br_from, overwrite)
146
# This should really never happen
150
dialog.info_dialog('%d revision(s) pushed.' % revs)
152
def close(self, widget=None):
153
self.window.destroy()
165
count = tree_to.pull(br_from, overwrite)