~ubuntu-branches/ubuntu/trusty/drizzle/trusty

« back to all changes in this revision

Viewing changes to plugin/pbms/src/backup_ms.cc

  • Committer: Bazaar Package Importer
  • Author(s): Monty Taylor
  • Date: 2010-10-02 14:17:48 UTC
  • mfrom: (1.1.1 upstream)
  • mto: (2.1.17 sid)
  • mto: This revision was merged to the branch mainline in revision 3.
  • Revision ID: james.westby@ubuntu.com-20101002141748-m6vbfbfjhrw1153e
Tags: 2010.09.1802-1
* New upstream release.
* Removed pid-file argument hack.
* Updated GPL-2 address to be new address.
* Directly copy in drizzledump.1 since debian doesn't have sphinx 1.0 yet.
* Link to jquery from libjs-jquery. Add it as a depend.
* Add drizzled.8 symlink to the install files.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/* Copyright (c) 2009 PrimeBase Technologies GmbH, Germany
 
2
 *
 
3
 * PrimeBase Media Stream for MySQL
 
4
 *
 
5
 * This program is free software; you can redistribute it and/or modify
 
6
 * it under the terms of the GNU General Public License as published by
 
7
 * the Free Software Foundation; either version 2 of the License, or
 
8
 * (at your option) any later version.
 
9
 *
 
10
 * This program is distributed in the hope that it will be useful,
 
11
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
12
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
13
 * GNU General Public License for more details.
 
14
 *
 
15
 * You should have received a copy of the GNU General Public License
 
16
 * along with this program; if not, write to the Free Software
 
17
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 
18
 *
 
19
 * Barry Leslie
 
20
 *
 
21
 * 2009-05-29
 
22
 *
 
23
 * H&G2JCtL
 
24
 *
 
25
 * Repository backup.
 
26
 *
 
27
 * The backup is done by creating a new database with the same name and ID in the 
 
28
 * backup location. Then the pbms_dump table in the source database is initialized
 
29
 * for a sequential scan for backup. This has the effect of locking all current repository
 
30
 * files. Then the equvalent of  'insert into dst_db.pbms_dump (select * from src_db.pbms_dump);'
 
31
 * is performed. 
 
32
 *
 
33
 */
 
34
 
 
35
#ifdef DRIZZLED
 
36
#include "config.h"
 
37
#include <drizzled/common.h>
 
38
#include <drizzled/session.h>
 
39
#include <drizzled/table.h>
 
40
#include <drizzled/message/table.pb.h>
 
41
#include "drizzled/charset_info.h"
 
42
#include <drizzled/table_proto.h>
 
43
#include <drizzled/field.h>
 
44
#include <drizzled/field/varstring.h>
 
45
#endif
 
46
 
 
47
#include "cslib/CSConfig.h"
 
48
 
 
49
#include <sys/types.h>
 
50
#include <inttypes.h>
 
51
 
 
52
#include "cslib/CSGlobal.h"
 
53
#include "cslib/CSStrUtil.h"
 
54
#include "cslib/CSStorage.h"
 
55
 
 
56
#include "defs_ms.h"
 
57
#include "system_table_ms.h"
 
58
#include "open_table_ms.h"
 
59
#include "table_ms.h"
 
60
#include "database_ms.h"
 
61
#include "repository_ms.h"
 
62
#include "backup_ms.h"
 
63
#include "transaction_ms.h"
 
64
#include "systab_variable_ms.h"
 
65
#include "systab_backup_ms.h"
 
66
 
 
67
uint32_t MSBackupInfo::gMaxInfoRef;
 
68
CSSyncSparseArray *MSBackupInfo::gBackupInfo;
 
69
 
 
70
//==========================================
 
71
MSBackupInfo::MSBackupInfo(     uint32_t id, 
 
72
                                                        const char *name, 
 
73
                                                        uint32_t db_id_arg, 
 
74
                                                        time_t start, 
 
75
                                                        time_t end, 
 
76
                                                        bool _isDump, 
 
77
                                                        const char *location, 
 
78
                                                        uint32_t cloudRef_arg, 
 
79
                                                        uint32_t cloudBackupNo_arg ):
 
80
        backupRefId(id),
 
81
        db_name(NULL),
 
82
        db_id(db_id_arg),
 
83
        startTime(start),
 
84
        completionTime(end),
 
85
        dump(_isDump),
 
86
        isRunning(false),
 
87
        backupLocation(NULL),
 
88
        cloudRef(cloudRef_arg),
 
89
        cloudBackupNo(cloudBackupNo_arg)
 
90
{
 
91
        db_name = CSString::newString(name);
 
92
        if (location && *location)              
 
93
                backupLocation = CSString::newString(location);         
 
94
}
 
95
 
 
96
//-------------------------------
 
97
MSBackupInfo::~MSBackupInfo()
 
98
{
 
99
        if (db_name)
 
100
                db_name->release();
 
101
        
 
102
        if (backupLocation)
 
103
                backupLocation->release();
 
104
}
 
105
 
 
106
//-------------------------------
 
107
void MSBackupInfo::startBackup(MSDatabase *pbms_db)
 
108
{
 
109
        MSDatabase *src_db;
 
110
        
 
111
        enter_();
 
112
        push_(pbms_db);
 
113
        
 
114
        src_db = MSDatabase::getDatabase(db_id);
 
115
        push_(src_db);
 
116
        
 
117
        startTime = time(NULL);
 
118
        
 
119
        src_db->startBackup(RETAIN(this));
 
120
        release_(src_db);
 
121
        
 
122
        isRunning = true;
 
123
        
 
124
        pop_(pbms_db);
 
125
        MSBackupTable::saveTable(pbms_db);
 
126
        exit_();
 
127
}
 
128
 
 
129
//-------------------------------
 
130
MSBackupInfo *MSBackupInfo::startDump(MSDatabase *db, uint32_t cloud_ref, uint32_t backup_no)
 
131
{
 
132
        MSBackupInfo *info;
 
133
        uint32_t ref_id;
 
134
        
 
135
        enter_();
 
136
        push_(db);
 
137
        lock_(gBackupInfo);
 
138
        
 
139
        ref_id = gMaxInfoRef++;
 
140
        new_(info, MSBackupInfo(ref_id, db->myDatabaseName->getCString(), db->myDatabaseID, time(NULL), 0, true, NULL, cloud_ref, backup_no));
 
141
        push_(info);
 
142
        
 
143
        gBackupInfo->set(ref_id, RETAIN(info));
 
144
        
 
145
        info->isRunning = true;
 
146
 
 
147
        pop_(info);
 
148
        unlock_(gBackupInfo);
 
149
        pop_(db);
 
150
        
 
151
        try_(a) {
 
152
                MSBackupTable::saveTable(db);
 
153
        }
 
154
        catch_(a);
 
155
        gBackupInfo->remove(ref_id);
 
156
        info->release();
 
157
        throw_();
 
158
        
 
159
        cont_(a);
 
160
        return_(info);
 
161
}
 
162
//-------------------------------
 
163
void MSBackupInfo::backupCompleted(MSDatabase *db)
 
164
{
 
165
        completionTime = time(NULL);    
 
166
        isRunning = false;
 
167
        MSBackupTable::saveTable(db);
 
168
}
 
169
 
 
170
//-------------------------------
 
171
void MSBackupInfo::backupTerminated(MSDatabase *db)
 
172
{
 
173
        enter_();
 
174
        push_(db);
 
175
        lock_(gBackupInfo);
 
176
        
 
177
        gBackupInfo->remove(backupRefId);
 
178
        unlock_(gBackupInfo);
 
179
        
 
180
        pop_(db);
 
181
        MSBackupTable::saveTable(db);
 
182
        exit_();
 
183
}
 
184
 
 
185
//==========================================
 
186
MSBackup::MSBackup():
 
187
CSDaemon(NULL),
 
188
bu_info(NULL),
 
189
bu_BackupList(NULL),
 
190
bu_Compactor(NULL),
 
191
bu_BackupRunning(false),
 
192
bu_State(BU_COMPLETED),
 
193
bu_SourceDatabase(NULL),
 
194
bu_Database(NULL),
 
195
bu_dst_dump(NULL),
 
196
bu_src_dump(NULL),
 
197
bu_size(0),
 
198
bu_completed(0),
 
199
bu_ID(0),
 
200
bu_start_time(0),
 
201
bu_TransactionManagerSuspended(false)
 
202
{
 
203
}
 
204
 
 
205
MSBackup *MSBackup::newMSBackup(MSBackupInfo *info)
 
206
{
 
207
        MSBackup *bu;
 
208
        enter_();
 
209
        
 
210
        push_(info);
 
211
        
 
212
        new_(bu, MSBackup());
 
213
        push_(bu);
 
214
        bu->bu_Database = MSDatabase::getBackupDatabase(RETAIN(info->backupLocation), RETAIN(info->db_name), info->db_id, true);
 
215
        pop_(bu);
 
216
        
 
217
        bu->bu_info = info;
 
218
        pop_(info);
 
219
 
 
220
        return_(bu);
 
221
}
 
222
 
 
223
void MSBackup::startBackup(MSDatabase *src_db)
 
224
{
 
225
        CSSyncVector    *repo_list;
 
226
        bool                    compacting = false;
 
227
        MSRepository    *repo;
 
228
        enter_();
 
229
 
 
230
        try_(a) {
 
231
                bu_SourceDatabase = src_db;
 
232
                repo_list = bu_SourceDatabase->getRepositoryList();
 
233
                // Suspend the compactor before locking the list.
 
234
                bu_Compactor = bu_SourceDatabase->getCompactorThread();
 
235
                if (bu_Compactor) {
 
236
                        bu_Compactor->retain();
 
237
                        bu_Compactor->suspend();
 
238
                }
 
239
 
 
240
                // Build the list of repositories to be backed up.
 
241
                lock_(repo_list);
 
242
 
 
243
                new_(bu_BackupList, CSVector(repo_list->size()));
 
244
                for (uint32_t i = 0; i<repo_list->size(); i++) {
 
245
                        if ((repo = (MSRepository *) repo_list->get(i))) {
 
246
                                if (!repo->isRemovingFP && !repo->mustBeDeleted) {
 
247
                                        bu_BackupList->add(RETAIN(repo));
 
248
                                        if (repo->initBackup() == REPO_COMPACTING) 
 
249
                                                compacting = true; 
 
250
                                        
 
251
                                        if (!repo->myRepoHeadSize) {
 
252
                                                /* The file has not yet been opened, so the
 
253
                                                 * garbage count will not be known!
 
254
                                                 */
 
255
                                                MSRepoFile *repo_file;
 
256
 
 
257
                                                //repo->retain();
 
258
                                                //unlock_(myRepostoryList);
 
259
                                                //push_(repo);
 
260
                                                repo_file = repo->openRepoFile();
 
261
                                                repo_file->release();
 
262
                                                //release_(repo);
 
263
                                                //lock_(myRepostoryList);
 
264
                                                //goto retry;
 
265
                                        }
 
266
                                        
 
267
                                        bu_size += repo->myRepoFileSize; 
 
268
 
 
269
                                }
 
270
                        }
 
271
                }
 
272
                
 
273
                // Copy the table list to the backup database:
 
274
                uint32_t                next_tab = 0;
 
275
                MSTable         *tab;
 
276
                while ((tab = bu_SourceDatabase->getNextTable(&next_tab))) {
 
277
                        push_(tab);
 
278
                        bu_Database->addTable(tab->myTableID, tab->myTableName->getCString(), 0, false);
 
279
                        release_(tab);
 
280
                }
 
281
                unlock_(repo_list);
 
282
                
 
283
                // Copy over any physical PBMS system tables.
 
284
                PBMSSystemTables::transferSystemTables(RETAIN(bu_Database), RETAIN(bu_SourceDatabase));
 
285
 
 
286
                // Load the system tables into the backup database. This will
 
287
                // initialize the database for cloud storage if required.
 
288
                PBMSSystemTables::loadSystemTables(RETAIN(bu_Database));
 
289
                
 
290
                // Set the cloud backup info.
 
291
                bu_Database->myBlobCloud->cl_setBackupInfo(RETAIN(bu_info));
 
292
                
 
293
                
 
294
                // Set the backup number in the pbms_variable tabe. (This is a hidden value.)
 
295
                // This value is used in case a drag and drop restore was done. When a data base is
 
296
                // first loaded this value is checked and if it is not zero then the backup record
 
297
                // will be read and any used to recover any BLOBs.
 
298
                // 
 
299
                char value[20];
 
300
                snprintf(value, 20, "%"PRIu32"", bu_info->getBackupRefId());
 
301
                MSVariableTable::setVariable(RETAIN(bu_Database), BACKUP_NUMBER_VAR, value);
 
302
                
 
303
                // Once the repositories are locked the compactor can be restarted
 
304
                // unless it is in the process of compacting a repository that is
 
305
                // being backed up.
 
306
                if (bu_Compactor && !compacting) {      
 
307
                        bu_Compactor->resume();         
 
308
                        bu_Compactor->release();                
 
309
                        bu_Compactor = NULL;            
 
310
                }
 
311
                
 
312
                // Suspend the transaction writer while the backup is running.
 
313
                MSTransactionManager::suspend(true);
 
314
                bu_TransactionManagerSuspended = true;
 
315
                
 
316
                // Start the backup daemon thread.
 
317
                bu_ID = bu_start_time = time(NULL);
 
318
                start();
 
319
        }
 
320
        catch_(a) {
 
321
                completeBackup();
 
322
                throw_();
 
323
        }
 
324
        cont_(a);
 
325
        
 
326
        exit_();
 
327
 
 
328
}
 
329
 
 
330
void MSBackup::completeBackup()
 
331
{
 
332
        if (bu_TransactionManagerSuspended) {   
 
333
                MSTransactionManager::resume();
 
334
                bu_TransactionManagerSuspended = false;
 
335
        }
 
336
 
 
337
        if (bu_BackupList) {
 
338
                MSRepository *repo;             
 
339
                
 
340
                while (bu_BackupList->size()) {
 
341
                        repo = (MSRepository *) bu_BackupList->take(0);
 
342
                        if (repo) {                             
 
343
                                repo->backupCompleted();
 
344
                                repo->release();                                
 
345
                        }
 
346
                }
 
347
                bu_BackupList->release();
 
348
                bu_BackupList = NULL;
 
349
        }
 
350
                
 
351
        if (bu_Compactor) {
 
352
                bu_Compactor->resume();
 
353
                bu_Compactor->release();
 
354
                bu_Compactor = NULL;
 
355
        }
 
356
        
 
357
        if (bu_Database) {
 
358
                if (bu_State == BU_COMPLETED)
 
359
                        bu_Database->releaseBackupDatabase();
 
360
                else 
 
361
                        MSDatabase::dropDatabase(bu_Database);
 
362
                        
 
363
                bu_Database = NULL;
 
364
        }
 
365
 
 
366
        if (bu_SourceDatabase){
 
367
                if (bu_State == BU_COMPLETED) 
 
368
                        bu_info->backupCompleted(bu_SourceDatabase);
 
369
                else 
 
370
                        bu_info->backupTerminated(bu_SourceDatabase);
 
371
                
 
372
                bu_SourceDatabase = NULL;
 
373
                bu_info->release();
 
374
                bu_info = NULL;
 
375
        }
 
376
        
 
377
        bu_BackupRunning = false;
 
378
}
 
379
 
 
380
bool MSBackup::doWork()
 
381
{
 
382
        CSMutex                         *my_lock;
 
383
        MSRepository            *src_repo, *dst_repo;
 
384
        MSRepoFile                      *src_file, *dst_file;
 
385
        off64_t                         src_offset, prev_offset;
 
386
        uint16_t                                head_size;
 
387
        uint64_t                                blob_size, blob_data_size;
 
388
        CSStringBuffer          *head;
 
389
        MSRepoPointersRec       ptr;
 
390
        uint32_t                                table_ref_count;
 
391
        uint32_t                                blob_ref_count;
 
392
        int                                     ref_count;
 
393
        size_t                          ref_size;
 
394
        uint32_t                                auth_code;
 
395
        uint32_t                                tab_id;
 
396
        uint64_t                                blob_id;
 
397
        MSOpenTable                     *otab;
 
398
        uint32_t                                src_repo_id;
 
399
        uint8_t                         status;
 
400
        uint8_t                         blob_storage_type;
 
401
        uint16_t                                tab_index;
 
402
        uint32_t                                mod_time;
 
403
        char                            transferBuffer[MS_BACKUP_BUFFER_SIZE];
 
404
        CloudKeyRec                     cloud_key;
 
405
 
 
406
        
 
407
        enter_();
 
408
        bu_BackupRunning = true;
 
409
        bu_State = BU_RUNNING; 
 
410
 
 
411
/*
 
412
        // For testing:
 
413
        {
 
414
                int blockit = 0;
 
415
                myWaitTime = 5 * 1000;  // Time in milli-seconds
 
416
                while (blockit)
 
417
                        return_(true);
 
418
        }
 
419
*/
 
420
        
 
421
        try_(a) {
 
422
                new_(head, CSStringBuffer(100));
 
423
                push_(head);
 
424
 
 
425
                src_repo = (MSRepository*)bu_BackupList->get(0);
 
426
                while (src_repo && !myMustQuit) {
 
427
                        src_offset = 0;
 
428
                        src_file = src_repo->openRepoFile();
 
429
                        push_(src_file);
 
430
 
 
431
                        dst_repo = bu_Database->lockRepo(src_repo->myRepoFileSize - src_repo->myGarbageCount);
 
432
                        frompool_(dst_repo);
 
433
                        dst_file = dst_repo->openRepoFile();
 
434
                        push_(dst_file);
 
435
                        
 
436
                        src_repo_id = src_repo->myRepoID;
 
437
                        src_offset = src_repo->myRepoHeadSize;
 
438
                        prev_offset = 0;
 
439
                        while (src_offset < src_repo->myRepoFileSize) { 
 
440
        retry_read:
 
441
                                        
 
442
                                bu_completed += src_offset - prev_offset;
 
443
                                prev_offset = src_offset;
 
444
                                suspended();
 
445
 
 
446
                                if (myMustQuit)
 
447
                                        break;
 
448
                                
 
449
                                // A lock is required here because references and dereferences to the
 
450
                                // BLOBs can result in the repository record being updated while 
 
451
                                // it is being copied.
 
452
                                my_lock = &src_repo->myRepoLock[src_offset % CS_REPO_REC_LOCK_COUNT];
 
453
                                lock_(my_lock);
 
454
                                head->setLength(src_repo->myRepoBlobHeadSize);
 
455
                                if (src_file->read(head->getBuffer(0), src_offset, src_repo->myRepoBlobHeadSize, 0) < src_repo->myRepoBlobHeadSize) { 
 
456
                                        unlock_(my_lock);
 
457
                                        break;
 
458
                                }
 
459
                                        
 
460
                                ptr.rp_chars = head->getBuffer(0);
 
461
                                ref_size = CS_GET_DISK_1(ptr.rp_head->rb_ref_size_1);
 
462
                                ref_count = CS_GET_DISK_2(ptr.rp_head->rb_ref_count_2);
 
463
                                head_size = CS_GET_DISK_2(ptr.rp_head->rb_head_size_2);
 
464
                                blob_size = CS_GET_DISK_6(ptr.rp_head->rb_blob_repo_size_6);
 
465
                                blob_data_size = CS_GET_DISK_6(ptr.rp_head->rb_blob_data_size_6);
 
466
                                auth_code = CS_GET_DISK_4(ptr.rp_head->rb_auth_code_4);
 
467
                                status = CS_GET_DISK_1(ptr.rp_head->rb_status_1);
 
468
                                mod_time = CS_GET_DISK_4(ptr.rp_head->rb_mod_time_4);
 
469
                                
 
470
                                blob_storage_type = CS_GET_DISK_1(ptr.rp_head->rb_storage_type_1);
 
471
                                if (blob_storage_type == MS_CLOUD_STORAGE) {
 
472
                                        MSRepoFile::getBlobKey(ptr.rp_head, &cloud_key);
 
473
                                }
 
474
 
 
475
                                // If the BLOB was modified after the start of the backup
 
476
                                // then set the mod time to the backup time to ensure that
 
477
                                // a backup for update will work correctly.
 
478
                                if (mod_time > bu_start_time)
 
479
                                        CS_SET_DISK_4(ptr.rp_head->rb_mod_time_4, bu_start_time);
 
480
                                        
 
481
                                // If the BLOB was moved during the time of this backup then copy
 
482
                                // it to the backup location as a referenced BLOB.
 
483
                                if ((status == MS_BLOB_MOVED)  && (bu_ID == (uint32_t) CS_GET_DISK_4(ptr.rp_head->rb_backup_id_4))) {
 
484
                                        status = MS_BLOB_REFERENCED;
 
485
                                        CS_SET_DISK_1(ptr.rp_head->rb_status_1, status);
 
486
                                }
 
487
                                
 
488
                                // sanity check
 
489
                                if ((blob_data_size == 0) || ref_count <= 0 || ref_size == 0 ||
 
490
                                        head_size < src_repo->myRepoBlobHeadSize + ref_count * ref_size ||
 
491
                                        !VALID_BLOB_STATUS(status)) {
 
492
                                        /* Can't be true. Assume this is garbage! */
 
493
                                        src_offset++;
 
494
                                        unlock_(my_lock);
 
495
                                        continue;
 
496
                                }
 
497
                                
 
498
                                
 
499
                                if ((status == MS_BLOB_REFERENCED) || (status == MS_BLOB_MOVED)) {
 
500
                                        head->setLength(head_size);
 
501
                                        if (src_file->read(head->getBuffer(0) + src_repo->myRepoBlobHeadSize, src_offset + src_repo->myRepoBlobHeadSize, head_size  - src_repo->myRepoBlobHeadSize, 0) != (head_size- src_repo->myRepoBlobHeadSize)) {
 
502
                                                unlock_(my_lock);
 
503
                                                break;
 
504
                                        }
 
505
 
 
506
                                        table_ref_count = 0;
 
507
                                        blob_ref_count = 0;
 
508
                                        
 
509
                                        // Loop through all the references removing temporary references 
 
510
                                        // and counting table and blob references.
 
511
                                        
 
512
                                        ptr.rp_chars = head->getBuffer(0) + src_repo->myRepoBlobHeadSize;
 
513
                                        for (int count = 0; count < ref_count; count++) {
 
514
                                                switch (CS_GET_DISK_2(ptr.rp_ref->rr_type_2)) {
 
515
                                                        case MS_BLOB_FREE_REF:
 
516
                                                                break;
 
517
                                                        case MS_BLOB_TABLE_REF:
 
518
                                                                // Unlike the compactor, table refs are not checked because
 
519
                                                                // they do not yet exist in the backup database.
 
520
                                                                table_ref_count++;
 
521
                                                                break;
 
522
                                                        case MS_BLOB_DELETE_REF:
 
523
                                                                // These are temporary references from the TempLog file. 
 
524
                                                                // They are not copied to the backup. 
 
525
                                                                CS_SET_DISK_2(ptr.rp_ref->rr_type_2, MS_BLOB_FREE_REF);
 
526
                                                                break;
 
527
                                                        default:
 
528
                                                                // Must be a BLOB reference
 
529
                                                                
 
530
                                                                tab_index = CS_GET_DISK_2(ptr.rp_blob_ref->er_table_2);
 
531
                                                                if (tab_index && (tab_index <= ref_count)) {
 
532
                                                                        // Only committed references are backed up.
 
533
                                                                        if (IS_COMMITTED(CS_GET_DISK_8(ptr.rp_blob_ref->er_blob_ref_id_8))) {
 
534
                                                                                MSRepoTableRefPtr       tab_ref;
 
535
                                                                                tab_ref = (MSRepoTableRefPtr) (head->getBuffer(0) + src_repo->myRepoBlobHeadSize + (tab_index-1) * ref_size);
 
536
                                                                                if (CS_GET_DISK_2(tab_ref->rr_type_2) == MS_BLOB_TABLE_REF)
 
537
                                                                                        blob_ref_count++;
 
538
                                                                        } else {
 
539
                                                                                CS_SET_DISK_2(ptr.rp_ref->rr_type_2, MS_BLOB_FREE_REF);
 
540
                                                                        }
 
541
                                                                
 
542
                                                                } else {
 
543
                                                                        /* Can't be true. Assume this is garbage! */
 
544
                                                                        src_offset++;
 
545
                                                                        unlock_(my_lock);
 
546
                                                                        goto retry_read;
 
547
                                                                }
 
548
                                                                break;
 
549
                                                }
 
550
                                                ptr.rp_chars += ref_size;
 
551
                                        }
 
552
 
 
553
 
 
554
                                        // If there are still blob references then the record needs to be backed up.
 
555
                                        if (table_ref_count && blob_ref_count) {
 
556
 
 
557
                                                off64_t dst_offset;
 
558
 
 
559
                                                dst_offset = dst_repo->myRepoFileSize;
 
560
                                                
 
561
                                                /* Write the header. */
 
562
                                                dst_file->write(head->getBuffer(0), dst_offset, head_size);
 
563
 
 
564
                                                /* Copy the BLOB over: */
 
565
                                                if (blob_storage_type == MS_CLOUD_STORAGE) { 
 
566
                                                        bu_Database->myBlobCloud->cl_backupBLOB(&cloud_key);
 
567
                                                } else
 
568
                                                        CSFile::transfer(dst_file, dst_offset + head_size, src_file, src_offset + head_size, blob_size, transferBuffer, MS_BACKUP_BUFFER_SIZE);
 
569
                                        
 
570
                                                /* Update the references: */
 
571
                                                ptr.rp_chars = head->getBuffer(0) + src_repo->myRepoBlobHeadSize;
 
572
                                                for (int count = 0; count < ref_count; count++) {
 
573
                                                        switch (CS_GET_DISK_2(ptr.rp_ref->rr_type_2)) {
 
574
                                                                case MS_BLOB_FREE_REF:
 
575
                                                                case MS_BLOB_DELETE_REF:
 
576
                                                                        break;
 
577
                                                                case MS_BLOB_TABLE_REF:
 
578
                                                                        tab_id = CS_GET_DISK_4(ptr.rp_tab_ref->tr_table_id_4);
 
579
                                                                        blob_id = CS_GET_DISK_6(ptr.rp_tab_ref->tr_blob_id_6);
 
580
 
 
581
                                                                        if ((otab = MSTableList::getOpenTableByID(bu_Database->myDatabaseID, tab_id))) {
 
582
                                                                                frompool_(otab);
 
583
                                                                                otab->getDBTable()->setBlobHandle(otab, blob_id, dst_repo->myRepoID, dst_offset, blob_size, head_size, auth_code);
 
584
//CSException::throwException(CS_CONTEXT, MS_ERR_NOT_IMPLEMENTED, "What if an error ocurred here!");
 
585
 
 
586
                                                                                backtopool_(otab);
 
587
                                                                        }
 
588
                                                                        break;
 
589
                                                                default:
 
590
                                                                        break;
 
591
                                                        }
 
592
                                                        ptr.rp_chars += ref_size;
 
593
                                                }
 
594
 
 
595
                                                dst_repo->myRepoFileSize += head_size + blob_size;
 
596
                                        }
 
597
                                }
 
598
                                unlock_(my_lock);
 
599
                                src_offset += head_size + blob_size;
 
600
                        }
 
601
                        bu_completed += src_offset - prev_offset;
 
602
                        
 
603
                        // close the destination repository and cleanup.
 
604
                        release_(dst_file);
 
605
                        backtopool_(dst_repo);
 
606
                        release_(src_file);
 
607
                        
 
608
                        // release the source repository and get the next one in the list.
 
609
                        src_repo->backupCompleted();
 
610
                        bu_BackupList->remove(0);
 
611
                        
 
612
                        src_repo = (MSRepository*)bu_BackupList->get(0);
 
613
                }
 
614
                                
 
615
                release_(head);
 
616
                if (myMustQuit)
 
617
                        bu_State = BU_TERMINATED; 
 
618
                else
 
619
                        bu_State = BU_COMPLETED; 
 
620
                        
 
621
        }       
 
622
        
 
623
        catch_(a) {
 
624
                logException();
 
625
        }
 
626
        
 
627
        cont_(a);       
 
628
        completeBackup();
 
629
        myMustQuit = true;
 
630
        return_(true);
 
631
}
 
632
 
 
633
void *MSBackup::completeWork()
 
634
{
 
635
        if (bu_SourceDatabase || bu_BackupList || bu_Compactor || bu_info) {
 
636
                // We shouldn't be here
 
637
                CSException::throwException(CS_CONTEXT, CS_ERR_GENERIC_ERROR, "MSBackup::completeBackup() not called");
 
638
                if (bu_SourceDatabase) {
 
639
                         bu_SourceDatabase->release();
 
640
                         bu_SourceDatabase = NULL;
 
641
                }
 
642
                        
 
643
                if (bu_BackupList) {
 
644
                         bu_BackupList->release();
 
645
                         bu_BackupList = NULL;
 
646
                }
 
647
 
 
648
                        
 
649
                if (bu_Compactor) {
 
650
                         bu_Compactor->release();
 
651
                         bu_Compactor = NULL;
 
652
                }
 
653
 
 
654
                        
 
655
                if (bu_info) {
 
656
                         bu_info->release();
 
657
                         bu_info = NULL;
 
658
                }
 
659
 
 
660
        }
 
661
        return NULL;
 
662
}