3 * wxbPanel for restoring files
5 * Nicolas Boichat, April-July 2004
10 Copyright (C) 2004 Kern Sibbald and John Walker
12 This program is free software; you can redistribute it and/or
13 modify it under the terms of the GNU General Public License
14 as published by the Free Software Foundation; either version 2
15 of the License, or (at your option) any later version.
17 This program is distributed in the hope that it will be useful,
18 but WITHOUT ANY WARRANTY; without even the implied warranty of
19 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 GNU General Public License for more details.
22 You should have received a copy of the GNU General Public License
23 along with this program; if not, write to the Free Software
24 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
26 /* Note concerning "done" output (modifiable marked with +)
28 +JobName: RestoreFiles
29 Bootstrap: /var/lib/bacula/restore.bsr
30 +Where: /tmp/bacula-restores
35 +When: 2004-04-18 01:18:56
37 OK to run? (yes/mod/no):mod
39 1: Level (not appropriate)
40 2: Storage (automatic ?)
43 5: Client (yes : "The defined Client resources are:\n\t1: velours-fd\n\t2: tom-fd\nSelect Client (File daemon) resource (1-2):")
44 6: When (yes : "Please enter desired start time as YYYY-MM-DD HH:MM:SS (return for now):")
45 7: Priority (yes : "Enter new Priority: (positive integer)")
47 9: Where (yes : "Please enter path prefix for restore (/ for none):")
48 10: Replace (yes : "Replace:\n 1: always\n 2: ifnewer\n 3: ifolder\n 4: never\n
49 Select replace option (1-4):")
51 Select parameter to modify (1-11):
54 #include "wxbrestorepanel.h"
56 #include "wxbmainframe.h"
60 #include <wx/choice.h>
61 #include <wx/datetime.h>
65 #include "unmarked.xpm"
67 #include "partmarked.xpm"
69 #include <wx/listimpl.cpp>
71 /* A macro named Yield is defined under MinGW */
74 WX_DEFINE_LIST(wxbEventList);
77 * Class which is stored in the tree and in the list to keep informations
80 class wxbTreeItemData : public wxTreeItemData {
82 wxbTreeItemData(wxString path, wxString name, int marked, long listid = -1);
83 wxbTreeItemData(wxString path, wxString name, wxString marked, long listid = -1);
89 void SetMarked(int marked);
90 void SetMarked(wxString marked);
94 static int GetMarkedStatus(wxString file);
96 wxString* path; /* Full path */
97 wxString* name; /* File name */
98 int marked; /* 0 - Not Marked, 1 - Marked, 2 - Some file under is marked */
99 long listid; /* list ID : >-1 if this data is in the list (and/or on the tree) */
102 wxbTreeItemData::wxbTreeItemData(wxString path, wxString name, int marked, long listid): wxTreeItemData() {
103 this->path = new wxString(path);
104 this->name = new wxString(name);
105 this->marked = marked;
106 this->listid = listid;
109 wxbTreeItemData::wxbTreeItemData(wxString path, wxString name, wxString marked, long listid): wxTreeItemData() {
110 this->path = new wxString(path);
111 this->name = new wxString(name);
113 this->listid = listid;
116 wxbTreeItemData::~wxbTreeItemData() {
121 int wxbTreeItemData::GetMarked() {
125 void wxbTreeItemData::SetMarked(wxString marked) {
129 else if (marked == "+") {
137 void wxbTreeItemData::SetMarked(int marked) {
138 this->marked = marked;
141 long wxbTreeItemData::GetListId() {
145 wxString wxbTreeItemData::GetPath() {
149 wxString wxbTreeItemData::GetName() {
153 /*wxbTreeItemData* wxbTreeItemData::GetChild(wxString dirname) {
154 int marked = GetMarkedStatus(dirname);
155 return new wxbTreeItemData(path + (marked ? dirname.Mid(1) : dirname), marked);
158 int wxbTreeItemData::GetMarkedStatus(wxString file) {
159 if (file.Length() == 0)
162 switch (file.GetChar(0)) {
172 // ----------------------------------------------------------------------------
173 // event tables and other macros for wxWindows
174 // ----------------------------------------------------------------------------
202 BEGIN_EVENT_TABLE(wxbRestorePanel, wxPanel)
203 EVT_BUTTON(RestoreStart, wxbRestorePanel::OnStart)
204 EVT_BUTTON(RestoreCancel, wxbRestorePanel::OnCancel)
206 EVT_TREE_SEL_CHANGING(TreeCtrl, wxbRestorePanel::OnTreeChanging)
207 EVT_TREE_SEL_CHANGED(TreeCtrl, wxbRestorePanel::OnTreeChanged)
208 EVT_TREE_ITEM_EXPANDING(TreeCtrl, wxbRestorePanel::OnTreeExpanding)
209 EVT_TREE_MARKED_EVENT(TreeCtrl, wxbRestorePanel::OnTreeMarked)
210 EVT_BUTTON(TreeAdd, wxbRestorePanel::OnTreeAdd)
211 EVT_BUTTON(TreeRemove, wxbRestorePanel::OnTreeRemove)
212 EVT_BUTTON(TreeRefresh, wxbRestorePanel::OnTreeRefresh)
214 EVT_LIST_ITEM_ACTIVATED(ListCtrl, wxbRestorePanel::OnListActivated)
215 EVT_LIST_MARKED_EVENT(ListCtrl, wxbRestorePanel::OnListMarked)
216 EVT_LIST_ITEM_SELECTED(ListCtrl, wxbRestorePanel::OnListChanged)
217 EVT_LIST_ITEM_DESELECTED(ListCtrl, wxbRestorePanel::OnListChanged)
218 EVT_BUTTON(ListAdd, wxbRestorePanel::OnListAdd)
219 EVT_BUTTON(ListRemove, wxbRestorePanel::OnListRemove)
220 EVT_BUTTON(ListRefresh, wxbRestorePanel::OnListRefresh)
222 EVT_TEXT(ConfigWhere, wxbRestorePanel::OnConfigUpdated)
223 EVT_TEXT(ConfigWhen, wxbRestorePanel::OnConfigUpdated)
224 EVT_TEXT(ConfigPriority, wxbRestorePanel::OnConfigUpdated)
225 EVT_CHOICE(ConfigWhen, wxbRestorePanel::OnConfigUpdated)
226 EVT_CHOICE(ConfigReplace, wxbRestorePanel::OnConfigUpdated)
227 EVT_CHOICE(ConfigClient, wxbRestorePanel::OnConfigUpdated)
228 EVT_CHOICE(ConfigFileset, wxbRestorePanel::OnConfigUpdated)
229 EVT_CHOICE(ConfigStorage, wxbRestorePanel::OnConfigUpdated)
230 EVT_CHOICE(ConfigJobName, wxbRestorePanel::OnConfigUpdated)
231 EVT_CHOICE(ConfigPool, wxbRestorePanel::OnConfigUpdated)
233 EVT_BUTTON(ConfigOk, wxbRestorePanel::OnConfigOk)
234 EVT_BUTTON(ConfigApply, wxbRestorePanel::OnConfigApply)
235 EVT_BUTTON(ConfigCancel, wxbRestorePanel::OnConfigCancel)
239 * wxbRestorePanel constructor
241 wxbRestorePanel::wxbRestorePanel(wxWindow* parent): wxbPanel(parent) {
242 //pendingEvents = new wxbEventList(); //EVTQUEUE
243 //processing = false; //EVTQUEUE
246 imagelist = new wxImageList(16, 16, TRUE, 3);
247 imagelist->Add(wxIcon(unmarked_xpm));
248 imagelist->Add(wxIcon(marked_xpm));
249 imagelist->Add(wxIcon(partmarked_xpm));
251 wxFlexGridSizer* mainSizer = new wxFlexGridSizer(3, 1, 10, 10);
252 mainSizer->AddGrowableCol(0);
253 mainSizer->AddGrowableRow(1);
255 wxFlexGridSizer *firstSizer = new wxFlexGridSizer(1, 2, 10, 10);
257 firstSizer->AddGrowableCol(0);
258 firstSizer->AddGrowableRow(0);
260 start = new wxButton(this, RestoreStart, "Enter restore mode", wxDefaultPosition, wxSize(150, 30));
261 firstSizer->Add(start, 0, wxALIGN_CENTER_HORIZONTAL | wxALIGN_CENTER_VERTICAL, 10);
263 cancel = new wxButton(this, RestoreCancel, "Cancel restore", wxDefaultPosition, wxSize(150, 30));
264 firstSizer->Add(cancel, 0, wxALIGN_CENTER_HORIZONTAL | wxALIGN_RIGHT, 10);
268 /* clientChoice = new wxChoice(this, ClientChoice, wxDefaultPosition, wxSize(150, 30), 0, elist);
269 firstSizer->Add(clientChoice, 1, wxALIGN_CENTER_HORIZONTAL | wxALIGN_CENTER_VERTICAL, 10);
271 jobChoice = new wxChoice(this, -1, wxDefaultPosition, wxSize(150, 30), 0, elist);
272 firstSizer->Add(jobChoice, 1, wxALIGN_CENTER_HORIZONTAL | wxALIGN_CENTER_VERTICAL, 10);*/
274 mainSizer->Add(firstSizer, 1, wxEXPAND, 10);
276 treelistPanel = new wxSplitterWindow(this);
278 wxPanel* treePanel = new wxPanel(treelistPanel);
279 wxFlexGridSizer *treeSizer = new wxFlexGridSizer(2, 1, 0, 0);
280 treeSizer->AddGrowableCol(0);
281 treeSizer->AddGrowableRow(0);
283 tree = new wxbTreeCtrl(treePanel, this, TreeCtrl, wxDefaultPosition, wxSize(200,50));
284 tree->SetImageList(imagelist);
286 treeSizer->Add(tree, 1, wxEXPAND, 0);
288 wxBoxSizer *treeCtrlSizer = new wxBoxSizer(wxHORIZONTAL);
289 treeadd = new wxButton(treePanel, TreeAdd, "Add", wxDefaultPosition, wxSize(60, 25));
290 treeCtrlSizer->Add(treeadd, 0, wxLEFT | wxRIGHT, 3);
291 treeremove = new wxButton(treePanel, TreeRemove, "Remove", wxDefaultPosition, wxSize(60, 25));
292 treeCtrlSizer->Add(treeremove, 0, wxLEFT | wxRIGHT, 3);
293 treerefresh = new wxButton(treePanel, TreeRefresh, "Refresh", wxDefaultPosition, wxSize(60, 25));
294 treeCtrlSizer->Add(treerefresh, 0, wxLEFT | wxRIGHT, 3);
296 treeSizer->Add(treeCtrlSizer, 1, wxALIGN_CENTER_HORIZONTAL, 0);
298 treePanel->SetSizer(treeSizer);
300 wxPanel* listPanel = new wxPanel(treelistPanel);
301 wxFlexGridSizer *listSizer = new wxFlexGridSizer(2, 1, 0, 0);
302 listSizer->AddGrowableCol(0);
303 listSizer->AddGrowableRow(0);
305 list = new wxbListCtrl(listPanel, this, ListCtrl, wxDefaultPosition, wxSize(200,50));
306 //treelistSizer->Add(list, 1, wxEXPAND, 10);
308 list->SetImageList(imagelist, wxIMAGE_LIST_SMALL);
311 info.SetMask(wxLIST_MASK_TEXT | wxLIST_MASK_FORMAT);
313 info.SetAlign(wxLIST_FORMAT_CENTER);
314 list->InsertColumn(0, info);
316 info.SetText("Filename");
317 info.SetAlign(wxLIST_FORMAT_LEFT);
318 list->InsertColumn(1, info);
320 info.SetText("Size");
321 info.SetAlign(wxLIST_FORMAT_RIGHT);
322 list->InsertColumn(2, info);
324 info.SetText("Date");
325 info.SetAlign(wxLIST_FORMAT_LEFT);
326 list->InsertColumn(3, info);
328 info.SetText("Perm.");
329 info.SetAlign(wxLIST_FORMAT_LEFT);
330 list->InsertColumn(4, info);
332 info.SetText("User");
333 info.SetAlign(wxLIST_FORMAT_RIGHT);
334 list->InsertColumn(5, info);
336 info.SetText("Group");
337 info.SetAlign(wxLIST_FORMAT_RIGHT);
338 list->InsertColumn(6, info);
340 listSizer->Add(list, 1, wxEXPAND, 0);
342 wxBoxSizer *listCtrlSizer = new wxBoxSizer(wxHORIZONTAL);
343 listadd = new wxButton(listPanel, ListAdd, "Add", wxDefaultPosition, wxSize(60, 25));
344 listCtrlSizer->Add(listadd, 0, wxLEFT | wxRIGHT, 5);
345 listremove = new wxButton(listPanel, ListRemove, "Remove", wxDefaultPosition, wxSize(60, 25));
346 listCtrlSizer->Add(listremove, 0, wxLEFT | wxRIGHT, 5);
347 listrefresh = new wxButton(listPanel, ListRefresh, "Refresh", wxDefaultPosition, wxSize(60, 25));
348 listCtrlSizer->Add(listrefresh, 0, wxLEFT | wxRIGHT, 5);
350 listSizer->Add(listCtrlSizer, 1, wxALIGN_CENTER_HORIZONTAL, 0);
352 listPanel->SetSizer(listSizer);
354 treelistPanel->SplitVertically(treePanel, listPanel, 210);
356 treelistPanel->SetMinimumPaneSize(210);
358 treelistPanel->Show(false);
360 wxbConfig* config = new wxbConfig();
361 config->Add(new wxbConfigParam("Job Name", ConfigJobName, choice, 0, elist));
362 config->Add(new wxbConfigParam("Client", ConfigClient, choice, 0, elist));
363 config->Add(new wxbConfigParam("Fileset", ConfigFileset, choice, 0, elist));
364 config->Add(new wxbConfigParam("Pool", ConfigPool, choice, 0, elist));
365 config->Add(new wxbConfigParam("Storage", ConfigStorage, choice, 0, elist));
366 config->Add(new wxbConfigParam("Before", ConfigWhen, choice, 0, elist));
368 configPanel = new wxbConfigPanel(this, config, "Please configure parameters concerning files to restore :", RestoreStart, RestoreCancel, -1);
370 configPanel->Show(true);
371 configPanel->Enable(false);
373 config = new wxbConfig();
374 config->Add(new wxbConfigParam("Job Name", -1, text, ""));
375 config->Add(new wxbConfigParam("Bootstrap", -1, text, ""));
376 config->Add(new wxbConfigParam("Where", ConfigWhere, modifiableText, ""));
377 wxString erlist[] = {"always", "if newer", "if older", "never"};
378 config->Add(new wxbConfigParam("Replace", ConfigReplace, choice, 4, erlist));
379 config->Add(new wxbConfigParam("Fileset", ConfigFileset, choice, 0, erlist));
380 config->Add(new wxbConfigParam("Client", ConfigClient, choice, 0, erlist));
381 config->Add(new wxbConfigParam("Storage", ConfigStorage, choice, 0, erlist));
382 config->Add(new wxbConfigParam("When", ConfigWhen, modifiableText, ""));
383 config->Add(new wxbConfigParam("Priority", ConfigPriority, modifiableText, ""));
385 restorePanel = new wxbConfigPanel(this, config, "Please configure parameters concerning files restoration :", ConfigOk, ConfigCancel, ConfigApply);
387 restorePanel->Show(false);
389 centerSizer = new wxBoxSizer(wxHORIZONTAL);
390 //centerSizer->Add(treelistPanel, 1, wxEXPAND | wxADJUST_MINSIZE);
392 mainSizer->Add(centerSizer, 1, wxEXPAND, 10);
394 gauge = new wxGauge(this, -1, 1, wxDefaultPosition, wxSize(200,20));
396 mainSizer->Add(gauge, 1, wxEXPAND, 5);
398 gauge->Enable(false);
401 mainSizer->SetSizeHints(this);
405 for (int i = 0; i < 7; i++) {
406 list->SetColumnWidth(i, 70);
409 SetCursor(*wxSTANDARD_CURSOR);
411 markWhenCommandDone = false;
417 * wxbRestorePanel destructor
419 wxbRestorePanel::~wxbRestorePanel() {
423 /*----------------------------------------------------------------------------
424 wxbPanel overloadings
425 ----------------------------------------------------------------------------*/
427 wxString wxbRestorePanel::GetTitle() {
431 void wxbRestorePanel::EnablePanel(bool enable) {
433 if (status == disabled) {
434 SetStatus(activable);
442 /*----------------------------------------------------------------------------
443 Commands called by events handler
444 ----------------------------------------------------------------------------*/
446 /* The main button has been clicked */
447 void wxbRestorePanel::CmdStart() {
449 if (status == activable) {
450 wxbMainFrame::GetInstance()->SetStatusText("Getting parameters list.");
451 wxbDataTokenizer* dt = wxbUtils::WaitForEnd(".clients\n", true, false);
454 configPanel->ClearRowChoices("Client");
455 restorePanel->ClearRowChoices("Client");
457 if (dt->GetCount() == 0) {
458 wxbMainFrame::GetInstance()->SetStatusText("Error : no clients returned by the director.");
462 for (i = 0; i < dt->GetCount(); i++) {
465 configPanel->AddRowChoice("Client", str);
466 restorePanel->AddRowChoice("Client", str);
476 dt = wxbUtils::WaitForEnd(".filesets\n", true, false);
478 configPanel->ClearRowChoices("Fileset");
479 restorePanel->ClearRowChoices("Fileset");
481 if (dt->GetCount() == 0) {
482 wxbMainFrame::GetInstance()->SetStatusText("Error : no filesets returned by the director.");
486 for (i = 0; i < dt->GetCount(); i++) {
489 configPanel->AddRowChoice("Fileset", str);
490 restorePanel->AddRowChoice("Fileset", str);
500 dt = wxbUtils::WaitForEnd(".storage\n", true, false);
502 configPanel->ClearRowChoices("Storage");
503 restorePanel->ClearRowChoices("Storage");
505 if (dt->GetCount() == 0) {
506 wxbMainFrame::GetInstance()->SetStatusText("Error : no storage returned by the director.");
510 for (i = 0; i < dt->GetCount(); i++) {
513 configPanel->AddRowChoice("Storage", str);
514 restorePanel->AddRowChoice("Storage", str);
524 dt = wxbUtils::WaitForEnd(".jobs\n", true, false);
526 configPanel->ClearRowChoices("Job Name");
528 if (dt->GetCount() == 0) {
529 wxbMainFrame::GetInstance()->SetStatusText("Error : no jobs returned by the director.");
533 for (i = 0; i < dt->GetCount(); i++) {
536 configPanel->AddRowChoice("Job Name", str);
539 configPanel->SetRowString("Job Name", "RestoreFiles");
548 dt = wxbUtils::WaitForEnd(".pools\n", true, false);
550 configPanel->ClearRowChoices("Pool");
552 if (dt->GetCount() == 0) {
553 wxbMainFrame::GetInstance()->SetStatusText("Error : no jobs returned by the director.");
557 for (i = 0; i < dt->GetCount(); i++) {
560 configPanel->AddRowChoice("Pool", str);
574 wxbMainFrame::GetInstance()->SetStatusText("Please configure your restore parameters.");
576 else if (status == entered) {
577 /* if (clientChoice->GetStringSelection().Length() < 1) {
578 wxbMainFrame::GetInstance()->SetStatusText("Please select a client.");
581 if (jobChoice->GetStringSelection().Length() < 1) {
582 wxbMainFrame::GetInstance()->SetStatusText("Please select a restore date.");
585 wxbMainFrame::GetInstance()->SetStatusText("Building restore tree...");
589 wxbTableParser* tableparser = new wxbTableParser();
590 wxbDataTokenizer* dt = new wxbDataTokenizer(false);
592 wxbMainFrame::GetInstance()->Send(wxString("restore") <<
593 " client=\"" << configPanel->GetRowString("Client") <<
594 "\" fileset=\"" << configPanel->GetRowString("Fileset") <<
595 "\" pool=\"" << configPanel->GetRowString("Pool") <<
596 "\" storage=\"" << configPanel->GetRowString("Storage") <<
597 "\" before=\"" << configPanel->GetRowString("Before") <<
599 //wxbUtils::WaitForPrompt("6\n");
601 /*wxbPromptParser *pp = wxbUtils::WaitForPrompt(wxString() << configPanel->GetRowString("Before") << "\n", true);
602 int client = pp->getChoices()->Index(configPanel->GetRowString("Client"));
603 if (client == wxNOT_FOUND) {
604 wxbMainFrame::GetInstance()->SetStatusText("Failed to find the selected client.");
609 //wxbMainFrame::GetInstance()->Send(wxString() << configPanel->GetRowString("Before") << "\n");
611 while (!tableparser->hasFinished() && !dt->hasFinished()) {
612 wxTheApp->Yield(true);
618 if (dt->hasFinished() && !tableparser->hasFinished()) {
620 if (dt->GetCount() > 1) {
621 str = (*dt)[dt->GetCount()-2];
624 wxbMainFrame::GetInstance()->SetStatusText(wxString("Error while starting restore: ") << str);
634 for (i = 0; i < tableparser->GetCount(); i++) {
635 str = (*tableparser)[i][2];
636 str.Replace(",", "");
637 if (str.ToLong(&l)) {
643 gauge->SetRange(tot);
645 /*wxbMainFrame::GetInstance()->Print(
646 wxString("[") << tot << "]", CS_DEBUG);*/
648 wxDateTime base = wxDateTime::Now();
652 unsigned int lastindex = 0;
659 newdate = wxDateTime::Now();
660 if (newdate.Subtract(base).GetMilliseconds() > 10 ) {
662 for (; lastindex < dt->GetCount(); lastindex++) {
663 if (((i1 = (*dt)[lastindex].Find("Building directory tree for JobId ")) >= 0) &&
664 ((i2 = (*dt)[lastindex].Find(" ...")) > 0)) {
665 str = (*dt)[lastindex].Mid(i1+34, i2-(i1+34));
666 for (i = 0; i < tableparser->GetCount(); i++) {
667 if (str == (*tableparser)[i][0]) {
668 str = (*tableparser)[i][2];
669 str.Replace(",", "");
670 if (str.ToLong(&l)) {
673 var = (willdo-done)/50;
674 gauge->SetValue(done);
675 wxTheApp->Yield(true);
681 else if ((*dt)[lastindex] == "+") {
682 gauge->SetValue(gauge->GetValue()+var);
683 wxTheApp->Yield(true);
688 if (dt->hasFinished()) {
692 wxTheApp->Yield(true);
696 gauge->SetValue(tot);
697 wxTheApp->Yield(true);
708 wxbUtils::WaitForEnd("unmark *\n");
709 wxTreeItemId root = tree->AddRoot(configPanel->GetRowString("Client"), -1, -1, new wxbTreeItemData("/", configPanel->GetRowString("Client"), 0));
710 currentTreeItem = root;
712 tree->SelectItem(root);
714 wxbMainFrame::GetInstance()->SetStatusText("Right click on a file or on a directory, or double-click on its mark to add it to the restore list.");
717 else if (status == choosing) {
721 wxbDataTokenizer* dt;
725 dt = new wxbDataTokenizer(true);
726 wxbUtils::WaitForPrompt("done\n");
728 SetStatus(configuring);
730 for (i = 0; i < dt->GetCount(); i++) {
731 if ((j = (*dt)[i].Find(" files selected to be restored.")) > -1) {
732 (*dt)[i].Mid(0, j).ToLong(&totfilemessages);
736 if ((j = (*dt)[i].Find(" file selected to be restored.")) > -1) {
737 (*dt)[i].Mid(0, j).ToLong(&totfilemessages);
742 wxbMainFrame::GetInstance()->SetStatusText(
743 wxString("Please configure your restore (")
744 << totfilemessages << " files selected to be restored)...");
746 UpdateSecondConfig(dt);
751 restorePanel->EnableApply(false);
753 if (totfilemessages == 0) {
754 wxbMainFrame::GetInstance()->Print("Restore failed : no file selected.\n", CS_DEBUG);
755 wxbMainFrame::GetInstance()->SetStatusText("Restore failed : no file selected.");
760 else if (status == configuring) {
761 cancel->Enable(false);
765 wxbMainFrame::GetInstance()->SetStatusText("Restoring, please wait...");
767 wxbDataTokenizer* dt;
769 SetStatus(restoring);
770 dt = wxbUtils::WaitForEnd("yes\n", true);
773 gauge->SetRange(totfilemessages);
777 for (i = 0; i < dt->GetCount(); i++) {
778 if ((j = (*dt)[i].Find("Job started. JobId=")) > -1) {
779 jobid = (*dt)[i].Mid(j+19);
780 wxbMainFrame::GetInstance()->SetStatusText("Restore started, jobid=" + jobid);
784 if ((j = (*dt)[i].Find("Job failed.")) > -1) {
785 wxbMainFrame::GetInstance()->Print("Restore failed, please look at messages.\n", CS_DEBUG);
786 wxbMainFrame::GetInstance()->SetStatusText("Restore failed, please look at messages in console.");
792 wxbMainFrame::GetInstance()->Print("Failed to retrieve jobid.\n", CS_DEBUG);
793 wxbMainFrame::GetInstance()->SetStatusText("Failed to retrieve jobid.\n");
797 wxDateTime currenttime;
799 dt = wxbUtils::WaitForEnd("time\n", true);
800 wxStringTokenizer ttkz((*dt)[0], " ", wxTOKEN_STRTOK);
801 if ((currenttime.ParseDate(ttkz.GetNextToken()) == NULL) || // Date
802 (currenttime.ParseTime(ttkz.GetNextToken()) == NULL)) { // Time
803 currenttime.SetYear(1990); // If parsing fails, set currenttime to a dummy date
806 currenttime -= wxTimeSpan::Seconds(30); //Adding a 30" tolerance
810 wxDateTime scheduledtime;
811 wxStringTokenizer stkz(restorePanel->GetRowString("When"), " ", wxTOKEN_STRTOK);
813 if ((scheduledtime.ParseDate(stkz.GetNextToken()) == NULL) || // Date
814 (scheduledtime.ParseTime(stkz.GetNextToken()) == NULL)) { // Time
815 scheduledtime.SetYear(2090); // If parsing fails, set scheduledtime to a dummy date
818 if (scheduledtime.Subtract(currenttime).IsLongerThan(wxTimeSpan::Seconds(150))) {
819 wxbMainFrame::GetInstance()->Print("Restore is scheduled in more than two minutes, wx-console will not wait for its completion.\n", CS_DEBUG);
820 wxbMainFrame::GetInstance()->SetStatusText("Restore is scheduled in more than two minutes, wx-console will not wait for its completion.");
825 wxString cmd = wxString("list jobid=") + jobid;
827 wxbTableParser* tableparser;
829 long filemessages = 0;
832 bool waitforever = false;
839 tableparser = wxbUtils::CreateAndWaitForParser(cmd);
841 status = (*tableparser)[0][7].GetChar(0);
844 wxbMainFrame::GetInstance()->SetStatusText("Restore job created, but not yet running.");
848 wxbMainFrame::GetInstance()->SetStatusText(
849 wxString("Restore job running, please wait (") << filemessages <<
850 " of " << totfilemessages << " files restored)...");
854 wxbMainFrame::GetInstance()->SetStatusText("Restore job terminated successfully.");
855 wxbMainFrame::GetInstance()->Print("Restore job terminated successfully.\n", CS_DEBUG);
859 case JS_ErrorTerminated:
860 wxbMainFrame::GetInstance()->SetStatusText("Restore job terminated in error, see messages in console.");
861 wxbMainFrame::GetInstance()->Print("Restore job terminated in error, see messages.\n", CS_DEBUG);
866 wxbMainFrame::GetInstance()->SetStatusText("Restore job reported a non-fatal error.");
870 wxbMainFrame::GetInstance()->SetStatusText("Restore job reported a fatal error.");
875 wxbMainFrame::GetInstance()->SetStatusText("Restore job cancelled by user.");
876 wxbMainFrame::GetInstance()->Print("Restore job cancelled by user.\n", CS_DEBUG);
881 wxbMainFrame::GetInstance()->SetStatusText("Restore job is waiting on File daemon.");
885 wxbMainFrame::GetInstance()->SetStatusText("Restore job is waiting for new media.");
888 case JS_WaitStoreRes:
889 wxbMainFrame::GetInstance()->SetStatusText("Restore job is waiting for storage resource.");
893 wxbMainFrame::GetInstance()->SetStatusText("Restore job is waiting for job resource.");
896 case JS_WaitClientRes:
897 wxbMainFrame::GetInstance()->SetStatusText("Restore job is waiting for Client resource.");
901 wxbMainFrame::GetInstance()->SetStatusText("Restore job is waiting for maximum jobs.");
904 case JS_WaitStartTime:
905 wxbMainFrame::GetInstance()->SetStatusText("Restore job is waiting for start time.");
908 case JS_WaitPriority:
909 wxbMainFrame::GetInstance()->SetStatusText("Restore job is waiting for higher priority jobs to finish.");
915 dt = wxbUtils::WaitForEnd(".messages\n", true);
917 for (unsigned int i = 0; i < dt->GetCount(); i++) {
918 wxStringTokenizer tkz((*dt)[i], " ", wxTOKEN_STRTOK);
922 // Date Time name: perm ? user grp size date time
923 //04-Apr-2004 17:19 Tom-fd: -rwx------ 1 nicolas None 514967 2004-03-20 20:03:42 filename
925 if (datetime.ParseDate(tkz.GetNextToken()) != NULL) { // Date
926 if (datetime.ParseTime(tkz.GetNextToken()) != NULL) { // Time
927 if (tkz.GetNextToken().Last() == ':') { // name:
928 tkz.GetNextToken(); // perm
929 tkz.GetNextToken(); // ?
930 tkz.GetNextToken(); // user
931 tkz.GetNextToken(); // grp
932 tkz.GetNextToken(); // size
933 if (datetime.ParseDate(tkz.GetNextToken()) != NULL) { //date
934 if (datetime.ParseTime(tkz.GetNextToken()) != NULL) { //time
936 //wxbMainFrame::GetInstance()->Print(wxString("(") << filemessages << ")", CS_DEBUG);
937 gauge->SetValue(filemessages);
948 while (sw2.Time() < 10000) {
949 wxTheApp->Yield(true);
957 if ((!waitforever) && (sw.Time() > 60000)) {
958 wxbMainFrame::GetInstance()->Print("The restore job has not been started within one minute, wx-console will not wait for its completion anymore.\n", CS_DEBUG);
959 wxbMainFrame::GetInstance()->SetStatusText("The restore job has not been started within one minute, wx-console will not wait for its completion anymore.");
964 wxbUtils::WaitForEnd(".messages\n");
966 gauge->SetValue(totfilemessages);
968 if (status == JS_Terminated) {
969 wxbMainFrame::GetInstance()->Print("Restore done successfully.\n", CS_DEBUG);
970 wxbMainFrame::GetInstance()->SetStatusText("Restore done successfully.");
976 /* The cancel button has been clicked */
977 void wxbRestorePanel::CmdCancel() {
980 if (status == restoring) {
982 wxbMainFrame::GetInstance()->Send(wxString("cancel job=") << jobid << "\n");
984 cancel->Enable(true);
989 while ((IsWorking()) && (cancelled != 2)) {
990 wxTheApp->Yield(true);
992 if (sw.Time() > 30000) { /* 30 seconds timeout */
993 if (status == choosing) {
994 wxbMainFrame::GetInstance()->Send("quit\n");
996 else if (status == configuring) {
997 wxbMainFrame::GetInstance()->Send("no\n");
999 else if (status == restoring) {
1002 SetStatus(finished);
1010 wxbMainFrame::GetInstance()->Send("quit\n");
1013 wxbMainFrame::GetInstance()->Send("no\n");
1019 SetStatus(finished);
1022 /* Apply configuration changes */
1024 /* 1: Level (not appropriate)
1029 * 6: When (yes : "Please enter desired start time as YYYY-MM-DD HH:MM:SS (return for now):")
1030 * 7: Priority (yes : "Enter new Priority: (positive integer)")
1032 * 9: Where (yes : "Please enter path prefix for restore (/ for none):")
1033 * 10: Replace (yes : "Replace:\n 1: always\n 2: ifnewer\n 3: ifolder\n 4: never\n
1034 * Select replace option (1-4):")
1038 void wxbRestorePanel::CmdConfigApply() {
1039 if (cfgUpdated == 0) return;
1041 wxbMainFrame::GetInstance()->SetStatusText("Applying restore configuration changes...");
1043 EnableConfig(false);
1045 wxbDataTokenizer* dt = NULL;
1047 bool failed = false;
1049 while (cfgUpdated > 0) {
1054 wxString def; //String to send if can't use our data
1055 if ((cfgUpdated >> ConfigWhere) & 1) {
1056 wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1057 wxbUtils::WaitForPrompt("9\n");
1058 dt = new wxbDataTokenizer(true);
1059 wxbUtils::WaitForPrompt(restorePanel->GetRowString("Where") + "\n");
1061 cfgUpdated = cfgUpdated & (~(1 << ConfigWhere));
1063 else if ((cfgUpdated >> ConfigReplace) & 1) {
1064 wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1065 wxbUtils::WaitForPrompt("10\n");
1066 dt = new wxbDataTokenizer(true);
1067 wxbUtils::WaitForPrompt(wxString() << (restorePanel->GetRowSelection("Replace")+1) << "\n");
1069 cfgUpdated = cfgUpdated & (~(1 << ConfigReplace));
1071 else if ((cfgUpdated >> ConfigWhen) & 1) {
1072 wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1073 wxbUtils::WaitForPrompt("6\n");
1074 dt = new wxbDataTokenizer(true);
1075 wxbUtils::WaitForPrompt(restorePanel->GetRowString("When") + "\n");
1077 cfgUpdated = cfgUpdated & (~(1 << ConfigWhen));
1079 else if ((cfgUpdated >> ConfigPriority) & 1) {
1080 wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1081 wxbUtils::WaitForPrompt("7\n");
1082 dt = new wxbDataTokenizer(true);
1083 wxbUtils::WaitForPrompt(restorePanel->GetRowString("Priority") + "\n");
1085 cfgUpdated = cfgUpdated & (~(1 << ConfigPriority));
1087 else if ((cfgUpdated >> ConfigClient) & 1) {
1088 wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1089 wxbPromptParser *pp = wxbUtils::WaitForPrompt("5\n", true);
1090 int client = pp->getChoices()->Index(restorePanel->GetRowString("Client"));
1091 if (client == wxNOT_FOUND) {
1092 wxbMainFrame::GetInstance()->SetStatusText("Failed to find the selected client.");
1097 dt = new wxbDataTokenizer(true);
1098 wxbUtils::WaitForPrompt(wxString() << client << "\n");
1100 cfgUpdated = cfgUpdated & (~(1 << ConfigClient));
1102 else if ((cfgUpdated >> ConfigFileset) & 1) {
1103 wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1104 wxbPromptParser *pp = wxbUtils::WaitForPrompt("4\n", true);
1105 int fileset = pp->getChoices()->Index(restorePanel->GetRowString("Fileset"));
1106 if (fileset == wxNOT_FOUND) {
1107 wxbMainFrame::GetInstance()->SetStatusText("Failed to find the selected fileset.");
1112 dt = new wxbDataTokenizer(true);
1113 wxbUtils::WaitForPrompt(wxString() << fileset << "\n");
1115 cfgUpdated = cfgUpdated & (~(1 << ConfigFileset));
1117 else if ((cfgUpdated >> ConfigStorage) & 1) {
1118 wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1119 wxbPromptParser *pp = wxbUtils::WaitForPrompt("2\n", true);
1120 int fileset = pp->getChoices()->Index(restorePanel->GetRowString("Storage"));
1121 if (fileset == wxNOT_FOUND) {
1122 wxbMainFrame::GetInstance()->SetStatusText("Failed to find the selected storage.");
1127 dt = new wxbDataTokenizer(true);
1128 wxbUtils::WaitForPrompt(wxString() << fileset << "\n");
1130 cfgUpdated = cfgUpdated & (~(1 << ConfigFileset));
1138 for (i = 0; i < dt->GetCount(); i++) {
1139 if ((*dt)[i].Find("Run Restore job") == 0) {
1144 if (i == dt->GetCount()) {
1146 dt = wxbUtils::WaitForEnd(def + "\n", true);
1150 UpdateSecondConfig(dt); /* TODO: Check result */
1155 wxbMainFrame::GetInstance()->SetStatusText("Restore configuration changes were applied.");
1161 /* Cancel restore */
1162 void wxbRestorePanel::CmdConfigCancel() {
1163 wxbUtils::WaitForEnd("no\n");
1164 wxbMainFrame::GetInstance()->Print("Restore cancelled.\n", CS_DEBUG);
1165 wxbMainFrame::GetInstance()->SetStatusText("Restore cancelled.");
1166 SetStatus(finished);
1169 /* List jobs for a specified client */
1170 void wxbRestorePanel::CmdListJobs() {
1171 if (status == entered) {
1172 configPanel->ClearRowChoices("Before");
1173 /*wxbUtils::WaitForPrompt("query\n");
1174 wxbUtils::WaitForPrompt("6\n");*/
1175 wxbTableParser* tableparser = new wxbTableParser(false);
1176 wxbDataTokenizer* dt = wxbUtils::WaitForEnd(
1177 wxString(".backups client=") + configPanel->GetRowString("Client") + "\n", true);
1179 while (!tableparser->hasFinished()) {
1180 wxTheApp->Yield(true);
1184 if (!tableparser->GetCount() == 0) {
1185 for (unsigned int i = 0; i < dt->Count(); i++) {
1186 if ((*dt)[i].Find("No results to list.") == 0) {
1187 configPanel->AddRowChoice("Before", "No backup found for this client.");
1188 configPanel->SetRowSelection("Before", 0);
1189 configPanel->EnableApply(true); // Enabling the not existing apply button disables the ok button.
1194 else if (((*dt)[i].Find("ERROR") > -1) || ((*dt)[i].Find("Query failed") > -1)) {
1195 configPanel->AddRowChoice("Before", "Cannot get previous backups list, see console.");
1196 configPanel->SetRowSelection("Before", 0);
1197 configPanel->EnableApply(true); // Enabling the not existing apply button disables the ok button.
1207 for (int i = tableparser->GetCount()-1; i > -1; i--) {
1208 wxString str = (*tableparser)[i][3];
1209 wxDateTime datetime;
1211 if ( ( (chr = datetime.ParseDate(str.GetData()) ) != NULL ) && ( datetime.ParseTime(++chr) != NULL ) ) {
1212 datetime += wxTimeSpan::Seconds(1);
1213 configPanel->AddRowChoice("Before", datetime.Format("%Y-%m-%d %H:%M:%S"));
1219 configPanel->SetRowSelection("Before", 0);
1220 configPanel->EnableApply(false); // Disabling the not existing apply button enables the ok button.
1224 /* List files and directories for a specified tree item */
1225 void wxbRestorePanel::CmdList(wxTreeItemId item) {
1226 if (status == choosing) {
1227 list->DeleteAllItems();
1232 UpdateTreeItem(item, true, false);
1234 if (list->GetItemCount() >= 1) {
1235 int firstwidth = list->GetSize().GetWidth();
1236 for (int i = 2; i < 7; i++) {
1237 list->SetColumnWidth(i, wxLIST_AUTOSIZE);
1238 firstwidth -= list->GetColumnWidth(i);
1241 list->SetColumnWidth(0, 18);
1243 list->SetColumnWidth(1, wxLIST_AUTOSIZE);
1244 if (list->GetColumnWidth(1) < firstwidth) {
1245 list->SetColumnWidth(1, firstwidth-25);
1251 /* Mark a treeitem (directory) or a listitem (file or directory) */
1252 void wxbRestorePanel::CmdMark(wxTreeItemId treeitem, long* listitems, int listsize, int state) {
1253 if (status == choosing) {
1254 wxbTreeItemData** itemdata;
1255 int itemdatasize = 0;
1256 if (listsize == 0) {
1257 itemdata = new wxbTreeItemData*[1];
1261 itemdata = new wxbTreeItemData*[listsize];
1262 itemdatasize = listsize;
1265 if (listitems != NULL) {
1266 for (int i = 0; i < listsize; i++) {
1267 itemdata[i] = (wxbTreeItemData*)list->GetItemData(listitems[i]);
1270 else if (treeitem.IsOk()) {
1271 itemdata[0] = (wxbTreeItemData*)tree->GetItemData(treeitem);
1278 if (itemdata[0] == NULL) { //Should never happen
1283 wxString dir = itemdata[0]->GetPath();
1287 if (dir.GetChar(dir.Length()-1) == '/') {
1291 int i = dir.Find('/', TRUE);
1296 else { /* first dir below root */
1297 file = dir.Mid(i+1);
1298 dir = dir.Mid(0, i+1);
1307 bool marked = false;
1308 bool unmarked = false;
1310 for (int i = 0; i < itemdatasize; i++) {
1311 switch(itemdata[i]->GetMarked()) {
1325 if (marked && unmarked)
1341 wxbUtils::WaitForEnd(wxString("cd \"") << dir << "\"\n");
1342 wxbUtils::WaitForEnd(wxString((state==1) ? "mark" : "unmark") << " \"" << file << "\"\n");
1344 /* TODO: Check commands results */
1346 /*if ((dir == "/") && (file == "*")) {
1347 itemdata->SetMarked((itemdata->GetMarked() == 1) ? 0 : 1);
1350 if (listitems == NULL) { /* tree item state changed */
1351 SetTreeItemState(treeitem, state);
1352 /*treeitem = tree->GetSelection();
1353 UpdateTree(treeitem, true);
1354 treeitem = tree->GetItemParent(treeitem);*/
1357 for (int i = 0; i < listsize; i++) {
1358 SetListItemState(listitems[i], state);
1360 /*UpdateTree(treeitem, (tree->GetSelection() == treeitem));
1361 treeitem = tree->GetItemParent(treeitem);*/
1364 /*while (treeitem.IsOk()) {
1365 WaitForList(treeitem, false);
1366 treeitem = tree->GetItemParent(treeitem);
1373 /*----------------------------------------------------------------------------
1375 ----------------------------------------------------------------------------*/
1377 /* Run a dir command, and waits until result is fully received. */
1378 void wxbRestorePanel::UpdateTreeItem(wxTreeItemId item, bool updatelist, bool recurse) {
1379 // this->updatelist = updatelist;
1380 wxbDataTokenizer* dt;
1382 dt = wxbUtils::WaitForEnd(wxString("cd \"") <<
1383 static_cast<wxbTreeItemData*>(tree->GetItemData(item))
1384 ->GetPath() << "\"\n", false);
1386 /* TODO: check command result */
1393 list->DeleteAllItems();
1394 dt = wxbUtils::WaitForEnd("dir\n", true);
1398 for (unsigned int i = 0; i < dt->GetCount(); i++) {
1401 if (str.Find("cwd is:") == 0) { // Sometimes cd command result "infiltrate" into listings.
1407 wxString* file = ParseList(str);
1412 wxTreeItemId treeid;
1414 if (file[8].GetChar(file[8].Length()-1) == '/') {
1418 treeid = tree->GetFirstChild(item, cookie);
1420 bool updated = false;
1422 while (treeid.IsOk()) {
1423 itemStr = tree->GetItemText(treeid);
1424 if (file[8] == itemStr) {
1425 int stat = wxbTreeItemData::GetMarkedStatus(file[6]);
1426 if (static_cast<wxbTreeItemData*>(tree->GetItemData(treeid))->GetMarked() != stat) {
1427 tree->SetItemImage(treeid, stat, wxTreeItemIcon_Normal);
1428 tree->SetItemImage(treeid, stat, wxTreeItemIcon_Selected);
1429 static_cast<wxbTreeItemData*>(tree->GetItemData(treeid))->SetMarked(file[6]);
1431 if ((recurse) && (tree->IsExpanded(treeid))) {
1432 UpdateTreeItem(treeid, false, true);
1437 treeid = tree->GetNextChild(item, cookie);
1441 int img = wxbTreeItemData::GetMarkedStatus(file[6]);
1442 treeid = tree->AppendItem(item, file[8], img, img, new wxbTreeItemData(file[7], file[8], file[6]));
1447 long ind = list->InsertItem(list->GetItemCount(), wxbTreeItemData::GetMarkedStatus(file[6]));
1448 wxbTreeItemData* data = new wxbTreeItemData(file[7], file[8], file[6], ind);
1449 data->SetId(treeid);
1450 list->SetItemData(ind, (long)data);
1451 list->SetItem(ind, 1, file[8]); // filename
1452 list->SetItem(ind, 2, file[4]); //Size
1453 list->SetItem(ind, 3, file[5]); //date
1454 list->SetItem(ind, 4, file[0]); //perm
1455 list->SetItem(ind, 5, file[2]); //user
1456 list->SetItem(ind, 6, file[3]); //grp
1468 /* Parse dir command results. */
1469 wxString* wxbRestorePanel::ParseList(wxString line) {
1470 /* See ls_output in dird/ua_tree.c */
1472 //drwxrwxrwx 1 root root 0 2004-04-03 14:35:21 f:/tocd/NVSU 1.00.00/
1473 //+ 10 + ++ + 8 + + 8 ++ 8 + + 19 + *+ ->
1474 //0 12 15 24 32 42 62
1476 if (line.Length() < 63)
1479 wxString* ret = new wxString[9];
1481 ret[0] = line.Mid(0, 10).Trim();
1482 ret[1] = line.Mid(12, 2).Trim();
1483 ret[2] = line.Mid(15, 8).Trim();
1484 ret[3] = line.Mid(24, 8).Trim();
1485 ret[4] = line.Mid(32, 8).Trim();
1486 ret[5] = line.Mid(42, 19).Trim();
1487 ret[6] = line.Mid(62, 1);
1488 ret[7] = line.Mid(63).Trim();
1490 if (ret[6] == " ") ret[6] = "";
1492 if (ret[7].GetChar(ret[7].Length()-1) == '/') {
1494 ret[8].RemoveLast();
1495 ret[8] = ret[7].Mid(ret[8].Find('/', true)+1);
1498 ret[8] = ret[7].Mid(ret[7].Find('/', true)+1);
1504 /* Sets a list item state, and update its parents and children if it is a directory */
1505 void wxbRestorePanel::SetListItemState(long listitem, int newstate) {
1506 wxbTreeItemData* itemdata = (wxbTreeItemData*)list->GetItemData(listitem);
1508 wxTreeItemId treeitem;
1510 itemdata->SetMarked(newstate);
1511 list->SetItemImage(listitem, newstate, 0); /* TODO: Find what these ints are for */
1512 list->SetItemImage(listitem, newstate, 1);
1514 if ((treeitem = itemdata->GetId()).IsOk()) {
1515 SetTreeItemState(treeitem, newstate);
1518 UpdateTreeItemState(tree->GetSelection());
1522 /* Sets a tree item state, and update its children, parents and list (if necessary) */
1523 void wxbRestorePanel::SetTreeItemState(wxTreeItemId item, int newstate) {
1525 wxTreeItemId currentChild = tree->GetFirstChild(item, cookie);
1527 wxbTreeItemData* itemdata;
1529 while (currentChild.IsOk()) {
1530 itemdata = (wxbTreeItemData*)tree->GetItemData(currentChild);
1531 int state = itemdata->GetMarked();
1533 if (state != newstate) {
1534 itemdata->SetMarked(newstate);
1535 tree->SetItemImage(currentChild, newstate, wxTreeItemIcon_Normal);
1536 tree->SetItemImage(currentChild, newstate, wxTreeItemIcon_Selected);
1539 currentChild = tree->GetNextChild(item, cookie);
1542 itemdata = (wxbTreeItemData*)tree->GetItemData(item);
1543 itemdata->SetMarked(newstate);
1544 tree->SetItemImage(item, newstate, wxTreeItemIcon_Normal);
1545 tree->SetItemImage(item, newstate, wxTreeItemIcon_Selected);
1548 if (tree->GetSelection() == item) {
1549 for (long i = 0; i < list->GetItemCount(); i++) {
1550 list->SetItemImage(i, newstate, 0); /* TODO: Find what these ints are for */
1551 list->SetItemImage(i, newstate, 1);
1555 UpdateTreeItemState(tree->GetItemParent(item));
1558 /* Update a tree item state, and its parents' state */
1559 void wxbRestorePanel::UpdateTreeItemState(wxTreeItemId item) {
1567 wxTreeItemId currentChild = tree->GetFirstChild(item, cookie);
1569 bool onechildmarked = false;
1570 bool onechildunmarked = false;
1572 while (currentChild.IsOk()) {
1573 state = ((wxbTreeItemData*)tree->GetItemData(currentChild))->GetMarked();
1576 onechildunmarked = true;
1579 onechildmarked = true;
1582 onechildmarked = true;
1583 onechildunmarked = true;
1587 if (onechildmarked && onechildunmarked) {
1591 currentChild = tree->GetNextChild(item, cookie);
1594 if (tree->GetSelection() == item) {
1595 for (long i = 0; i < list->GetItemCount(); i++) {
1596 state = ((wxbTreeItemData*)list->GetItemData(i))->GetMarked();
1600 onechildunmarked = true;
1603 onechildmarked = true;
1606 onechildmarked = true;
1607 onechildunmarked = true;
1611 if (onechildmarked && onechildunmarked) {
1619 if (onechildmarked && onechildunmarked) {
1622 else if (onechildmarked) {
1625 else if (onechildunmarked) {
1628 else { // no child, don't change anything
1629 UpdateTreeItemState(tree->GetItemParent(item));
1633 wxbTreeItemData* itemdata = (wxbTreeItemData*)tree->GetItemData(item);
1635 itemdata->SetMarked(state);
1636 tree->SetItemImage(item, state, wxTreeItemIcon_Normal);
1637 tree->SetItemImage(item, state, wxTreeItemIcon_Selected);
1639 UpdateTreeItemState(tree->GetItemParent(item));
1642 /* Refresh the whole tree. */
1643 void wxbRestorePanel::RefreshTree() {
1644 /* Save current selection */
1645 wxArrayString current;
1647 wxTreeItemId item = currentTreeItem;
1649 while ((item.IsOk()) && (item != tree->GetRootItem())) {
1650 current.Add(tree->GetItemText(item));
1651 item = tree->GetItemParent(item);
1654 /* Update the tree */
1655 UpdateTreeItem(tree->GetRootItem(), false, true);
1657 /* Reselect the former selected item */
1658 item = tree->GetRootItem();
1660 if (current.Count() == 0) {
1661 tree->SelectItem(item);
1667 for (int i = current.Count()-1; i >= 0; i--) {
1669 wxTreeItemId currentChild = tree->GetFirstChild(item, cookie);
1673 while (currentChild.IsOk()) {
1674 if (tree->GetItemText(currentChild) == current[i]) {
1675 item = currentChild;
1680 currentChild = tree->GetNextChild(item, cookie);
1686 UpdateTreeItem(item, true, false); /* Update the list */
1688 tree->SelectItem(item);
1691 void wxbRestorePanel::RefreshList() {
1692 if (currentTreeItem.IsOk()) {
1693 UpdateTreeItem(currentTreeItem, true, false); /* Update the list */
1697 /* Update first config, adapting settings to the job name selected */
1698 void wxbRestorePanel::UpdateFirstConfig() {
1699 configPanel->Enable(false);
1700 wxbDataTokenizer* dt = wxbUtils::WaitForEnd(wxString(".defaults job=") + configPanel->GetRowString("Job Name") + "\n", true, false);
1706 * where=/tmp/bacula-restores
1709 * fileset=Full Set */
1715 bool dolistjobs = false;
1717 for (i = 0; i < dt->GetCount(); i++) {
1719 if ((j = str.Find('=')) > -1) {
1720 name = str.Mid(0, j);
1721 if (name == "pool") {
1722 configPanel->SetRowString("Pool", str.Mid(j+1));
1724 else if (name == "client") {
1726 if ((str != configPanel->GetRowString("Client")) || (configPanel->GetRowString("Before") == "")) {
1727 configPanel->SetRowString("Client", str);
1731 else if (name == "storage") {
1732 configPanel->SetRowString("Storage", str.Mid(j+1));
1734 else if (name == "fileset") {
1735 configPanel->SetRowString("Fileset", str.Mid(j+1));
1743 //wxTheApp->Yield(false);
1746 configPanel->Enable(true);
1750 * Update second config.
1753 * JobName: RestoreFiles
1754 * Bootstrap: /var/lib/bacula/restore.bsr
1755 * Where: /tmp/bacula-restores
1760 * When: 2004-04-18 01:18:56
1762 * OK to run? (yes/mod/no):
1765 bool wxbRestorePanel::UpdateSecondConfig(wxbDataTokenizer* dt) {
1767 for (i = 0; i < dt->GetCount(); i++) {
1768 if ((*dt)[i].Find("Run Restore job") == 0)
1772 if ((i + 10) > dt->GetCount()) {
1778 if ((k = (*dt)[++i].Find("JobName:")) != 0) return false;
1779 restorePanel->SetRowString("Job Name", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1780 if ((k = (*dt)[++i].Find("Bootstrap:")) != 0) return false;
1781 restorePanel->SetRowString("Bootstrap", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1782 if ((k = (*dt)[++i].Find("Where:")) != 0) return false;
1783 restorePanel->SetRowString("Where", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1785 if ((k = (*dt)[++i].Find("Replace:")) != 0) return false;
1786 wxString str = (*dt)[i].Mid(10).Trim(false).RemoveLast();
1787 if (str == "always") restorePanel->SetRowSelection("Replace", 0);
1788 else if (str == "ifnewer") restorePanel->SetRowSelection("Replace", 1);
1789 else if (str == "ifolder") restorePanel->SetRowSelection("Replace", 2);
1790 else if (str == "never") restorePanel->SetRowSelection("Replace", 3);
1791 else restorePanel->SetRowSelection("Replace", 0);
1793 if ((k = (*dt)[++i].Find("FileSet:")) != 0) return false;
1794 restorePanel->SetRowString("Fileset", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1795 if ((k = (*dt)[++i].Find("Client:")) != 0) return false;
1796 restorePanel->SetRowString("Client", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1797 if ((k = (*dt)[++i].Find("Storage:")) != 0) return false;
1798 restorePanel->SetRowString("Storage", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1799 if ((k = (*dt)[++i].Find("When:")) != 0) return false;
1800 restorePanel->SetRowString("When", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1801 if ((k = (*dt)[++i].Find("Priority:")) != 0) return false;
1802 restorePanel->SetRowString("Priority", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1805 restorePanel->Layout();
1810 /*----------------------------------------------------------------------------
1812 ----------------------------------------------------------------------------*/
1814 /* Set current status by enabling/disabling components */
1815 void wxbRestorePanel::SetStatus(status_enum newstatus) {
1816 switch (newstatus) {
1818 centerSizer->Remove(configPanel);
1819 centerSizer->Remove(restorePanel);
1820 centerSizer->Remove(treelistPanel);
1821 treelistPanel->Show(false);
1822 restorePanel->Show(false);
1823 centerSizer->Add(configPanel, 1, wxEXPAND);
1824 configPanel->Show(true);
1825 configPanel->Layout();
1826 centerSizer->Layout();
1828 start->SetLabel("Enter restore mode");
1829 start->Enable(false);
1830 configPanel->Enable(false);
1831 tree->Enable(false);
1832 list->Enable(false);
1833 gauge->Enable(false);
1834 cancel->Enable(false);
1839 centerSizer->Remove(configPanel);
1840 centerSizer->Remove(restorePanel);
1841 centerSizer->Remove(treelistPanel);
1842 treelistPanel->Show(false);
1843 restorePanel->Show(false);
1844 centerSizer->Add(configPanel, 1, wxEXPAND);
1845 configPanel->Show(true);
1846 configPanel->Layout();
1847 centerSizer->Layout();
1849 tree->DeleteAllItems();
1850 list->DeleteAllItems();
1851 configPanel->ClearRowChoices("Client");
1852 configPanel->ClearRowChoices("Before");
1853 wxbMainFrame::GetInstance()->EnablePanels();
1854 newstatus = activable;
1857 start->SetLabel("Enter restore mode");
1858 start->Enable(true);
1859 configPanel->Enable(false);
1860 tree->Enable(false);
1861 list->Enable(false);
1862 gauge->Enable(false);
1863 cancel->Enable(false);
1867 wxbMainFrame::GetInstance()->DisablePanels(this);
1869 start->Enable(false);
1870 //start->SetLabel("Choose files to restore");
1871 configPanel->Enable(true);
1872 tree->Enable(false);
1873 list->Enable(false);
1874 cancel->Enable(true);
1881 start->Enable(true);
1882 start->SetLabel("Restore");
1883 centerSizer->Remove(configPanel);
1884 configPanel->Show(false);
1885 centerSizer->Add(treelistPanel, 1, wxEXPAND);
1886 treelistPanel->Show(true);
1887 treelistPanel->Layout();
1888 centerSizer->Layout();
1895 start->Enable(false);
1896 configPanel->Enable(false);
1897 tree->Enable(false);
1898 list->Enable(false);
1899 centerSizer->Remove(treelistPanel);
1900 treelistPanel->Show(false);
1901 centerSizer->Add(restorePanel, 1, wxEXPAND);
1902 restorePanel->Show(true);
1903 restorePanel->Layout();
1904 centerSizer->Layout();
1906 restorePanel->EnableApply(false);
1909 start->SetLabel("Restoring...");
1910 gauge->Enable(true);
1912 start->Enable(false);
1913 configPanel->Enable(false);
1914 tree->Enable(false);
1915 list->Enable(false);
1922 /*----------------------------------------------------------------------------
1924 ----------------------------------------------------------------------------*/
1926 void wxbRestorePanel::SetWorking(bool working) {
1927 this->working = working;
1929 SetCursor(*wxHOURGLASS_CURSOR);
1930 // SetEvtHandlerEnabled(false); //EVTQUEUE
1932 // else if (!processing) { /* Empty event queue if we aren't already doing this */ //EVTQUEUE
1934 // processing = true; //EVTQUEUE
1935 SetCursor(*wxSTANDARD_CURSOR);
1936 // SetEvtHandlerEnabled(true); //EVTQUEUE
1937 /* wxNode *node = pendingEvents->First(); //EVTQUEUE
1939 wxEvent *event = (wxEvent *)node->Data();
1942 wxEvtHandler::ProcessEvent(*event);
1945 node = pendingEvents->First();
1947 processing = false;*/
1951 bool wxbRestorePanel::IsWorking() {
1952 return this->working;
1955 void wxbRestorePanel::EnableConfig(bool enable) {
1956 restorePanel->Enable(enable);
1959 /*----------------------------------------------------------------------------
1961 ----------------------------------------------------------------------------*/
1966 bool wxbRestorePanel::ProcessEvent(wxEvent& event) {
1967 if (IsWorking() || processing) {
1968 wxEvent *eventCopy = event.Clone();
1970 pendingEvents->Append(eventCopy);
1974 return wxEvtHandler::ProcessEvent(event);
1979 void wxbRestorePanel::OnCancel(wxCommandEvent& event) {
1980 cancel->Enable(false);
1981 SetCursor(*wxHOURGLASS_CURSOR);
1983 SetCursor(*wxSTANDARD_CURSOR);
1986 void wxbRestorePanel::OnStart(wxCommandEvent& event) {
1995 void wxbRestorePanel::OnTreeChanging(wxTreeEvent& event) {
2001 void wxbRestorePanel::OnTreeExpanding(wxTreeEvent& event) {
2007 //CmdList(event.GetItem());
2008 if (tree->GetSelection() != event.GetItem()) {
2009 tree->SelectItem(event.GetItem());
2014 void wxbRestorePanel::OnTreeChanged(wxTreeEvent& event) {
2018 if (currentTreeItem == event.GetItem()) {
2021 treeadd->Enable(false);
2022 treeremove->Enable(false);
2023 treerefresh->Enable(false);
2024 markWhenCommandDone = false;
2026 currentTreeItem = event.GetItem();
2027 CmdList(event.GetItem());
2028 if (markWhenCommandDone) {
2029 CmdMark(event.GetItem(), NULL, 0);
2033 if (event.GetItem().IsOk()) {
2034 int status = ((wxbTreeItemData*)tree->GetItemData(event.GetItem()))->GetMarked();
2035 treeadd->Enable(status != 1);
2036 treeremove->Enable(status != 0);
2038 treerefresh->Enable(true);
2041 void wxbRestorePanel::OnTreeMarked(wxbTreeMarkedEvent& event) {
2043 if (tree->GetSelection() == event.GetItem()) {
2044 markWhenCommandDone = !markWhenCommandDone;
2049 markWhenCommandDone = false;
2050 CmdMark(event.GetItem(), NULL, 0);
2051 if (markWhenCommandDone) {
2052 CmdMark(event.GetItem(), NULL, 0);
2057 if (event.GetItem().IsOk()) {
2058 int status = ((wxbTreeItemData*)tree->GetItemData(event.GetItem()))->GetMarked();
2059 treeadd->Enable(status != 1);
2060 treeremove->Enable(status != 0);
2064 void wxbRestorePanel::OnTreeAdd(wxCommandEvent& event) {
2069 if (currentTreeItem.IsOk()) {
2071 CmdMark(currentTreeItem, NULL, 0, 1);
2074 treeremove->Enable(1);
2079 void wxbRestorePanel::OnTreeRemove(wxCommandEvent& event) {
2084 if (currentTreeItem.IsOk()) {
2086 CmdMark(currentTreeItem, NULL, 0, 0);
2089 treeremove->Enable(0);
2094 void wxbRestorePanel::OnTreeRefresh(wxCommandEvent& event) {
2104 void wxbRestorePanel::OnListMarked(wxbListMarkedEvent& event) {
2109 if (list->GetSelectedItemCount() == 0) {
2115 long* items = new long[list->GetSelectedItemCount()];
2119 long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2120 while (item != -1) {
2123 item = list->GetNextItem(item, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2126 CmdMark(wxTreeItemId(), items, num);
2130 wxListEvent listevt;
2132 OnListChanged(listevt);
2139 void wxbRestorePanel::OnListActivated(wxListEvent& event) {
2144 long item = event.GetIndex();
2145 // long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_FOCUSED);
2147 wxbTreeItemData* itemdata = (wxbTreeItemData*)list->GetItemData(item);
2148 wxString name = itemdata->GetName();
2155 if (name.GetChar(name.Length()-1) == '/') {
2156 wxTreeItemId currentChild = tree->GetFirstChild(currentTreeItem, cookie);
2158 while (currentChild.IsOk()) {
2159 wxString name2 = tree->GetItemText(currentChild);
2160 if (name2 == name) {
2161 //tree->UnselectAll();
2163 tree->Expand(currentTreeItem);
2164 tree->SelectItem(currentChild);
2168 currentChild = tree->GetNextChild(currentTreeItem, cookie);
2175 void wxbRestorePanel::OnListChanged(wxListEvent& event) {
2180 listadd->Enable(false);
2181 listremove->Enable(false);
2183 bool marked = false;
2184 bool unmarked = false;
2186 long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2187 while (item != -1) {
2188 switch (((wxbTreeItemData*)list->GetItemData(item))->GetMarked()) {
2201 // Should never happen
2203 if (marked && unmarked) break;
2204 item = list->GetNextItem(item, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2207 listadd->Enable(unmarked);
2208 listremove->Enable(marked);
2211 void wxbRestorePanel::OnListAdd(wxCommandEvent& event) {
2218 long* items = new long[list->GetSelectedItemCount()];
2222 long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2223 while (item != -1) {
2226 item = list->GetNextItem(item, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2229 CmdMark(wxTreeItemId(), items, num, 1);
2236 listadd->Enable(false);
2237 listremove->Enable(true);
2240 void wxbRestorePanel::OnListRemove(wxCommandEvent& event) {
2247 long* items = new long[list->GetSelectedItemCount()];
2251 long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2252 while (item != -1) {
2255 item = list->GetNextItem(item, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2258 CmdMark(wxTreeItemId(), items, num, 0);
2265 listadd->Enable(true);
2266 listremove->Enable(false);
2269 void wxbRestorePanel::OnListRefresh(wxCommandEvent& event) {
2279 void wxbRestorePanel::OnConfigUpdated(wxCommandEvent& event) {
2280 if (status == entered) {
2281 if (event.GetId() == ConfigJobName) {
2286 UpdateFirstConfig();
2289 else if (event.GetId() == ConfigClient) {
2294 configPanel->Enable(false);
2296 configPanel->Enable(true);
2299 cfgUpdated = cfgUpdated | (1 << event.GetId());
2301 else if (status == configuring) {
2302 restorePanel->EnableApply(true);
2303 cfgUpdated = cfgUpdated | (1 << event.GetId());
2307 void wxbRestorePanel::OnConfigOk(wxCommandEvent& WXUNUSED(event)) {
2308 if (status != configuring) return;
2317 void wxbRestorePanel::OnConfigApply(wxCommandEvent& WXUNUSED(event)) {
2318 if (status != configuring) return;
2324 if (cfgUpdated == 0) {
2325 restorePanel->EnableApply(false);
2330 void wxbRestorePanel::OnConfigCancel(wxCommandEvent& WXUNUSED(event)) {
2331 if (status != configuring) return;