mirror of
https://github.com/duplicati/duplicati.git
synced 2025-11-28 03:20:25 +08:00
456 lines
19 KiB
C#
456 lines
19 KiB
C#
// Copyright (C) 2025, The Duplicati Team
|
|
// https://duplicati.com, hello@duplicati.com
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
// to deal in the Software without restriction, including without limitation
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
using System;
|
|
using System.Threading.Tasks;
|
|
using Duplicati.Library.Main.Database;
|
|
using CoCoL;
|
|
using Duplicati.Library.Main.Operation.Common;
|
|
using System.Collections.Generic;
|
|
using Duplicati.Library.Main.Volumes;
|
|
using System.Linq;
|
|
using Duplicati.Library.Interface;
|
|
using System.Threading;
|
|
|
|
namespace Duplicati.Library.Main.Operation.Backup
|
|
{
|
|
/// <summary>
|
|
/// Asynchronous interface that ensures all requests
|
|
/// to the database are performed in a sequential manner
|
|
/// </summary>
|
|
internal class BackupDatabase : DatabaseCommon
|
|
{
|
|
private readonly LocalBackupDatabase m_database;
|
|
|
|
public class FileEntryData
|
|
{
|
|
public long id;
|
|
public DateTime modified;
|
|
public long filesize;
|
|
public string metahash;
|
|
public long metasize;
|
|
}
|
|
|
|
public BackupDatabase(LocalBackupDatabase database, Options options)
|
|
: base(database, options)
|
|
{
|
|
m_database = database;
|
|
}
|
|
|
|
public Task<long> FindBlockIDAsync(string hash, long size, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.FindBlockID(hash, size, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<bool> AddBlockAsync(string hash, long size, long volumeid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AddBlock(hash, size, volumeid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<string> GetFileHashAsync(long fileid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.GetFileHash(fileid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<(bool, long)> AddMetadatasetAsync(string hash, long size, long blocksetid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AddMetadataset(hash, size, blocksetid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<(bool, long)> GetMetadataIDAsync(string hash, long size, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.GetMetadatasetID(hash, size, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task AddDirectoryEntryAsync(string filename, long metadataid, DateTime lastModified, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AddDirectoryEntry(filename, metadataid, lastModified, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task AddSymlinkEntryAsync(string filename, long metadataid, DateTime lastModified, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AddSymlinkEntry(filename, metadataid, lastModified, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<(string MetadataHash, long Size)?> GetMetadataHashAndSizeForFileAsync(long fileid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.GetMetadataHashAndSizeForFile(fileid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<(long, DateTime, long)> GetFileLastModifiedAsync(long prefixid, string path, long lastfilesetid, bool includeLength, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.GetFileLastModified(prefixid, path, lastfilesetid, includeLength, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<FileEntryData> GetFileEntryAsync(long prefixid, string path, long lastfilesetid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
{
|
|
var (id, oldModified, lastFileSize, oldMetahash, oldMetasize) =
|
|
await m_database.GetFileEntry(prefixid, path, lastfilesetid, cancellationToken)
|
|
.ConfigureAwait(false);
|
|
|
|
return
|
|
id < 0 ?
|
|
null
|
|
:
|
|
new FileEntryData()
|
|
{
|
|
id = id,
|
|
modified = oldModified,
|
|
filesize = lastFileSize,
|
|
metahash = oldMetahash,
|
|
metasize = oldMetasize
|
|
};
|
|
});
|
|
}
|
|
|
|
public Task<long> AddBlocksetAsync(string filehash, long size, int blocksize, IEnumerable<string> hashlist, IEnumerable<string> blocklisthashes, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
{
|
|
var (_, blocksetid) = await m_database
|
|
.AddBlockset(filehash, size, blocksize, hashlist, blocklisthashes, cancellationToken)
|
|
.ConfigureAwait(false);
|
|
|
|
return blocksetid;
|
|
});
|
|
}
|
|
|
|
public Task<long> GetOrCreatePathPrefix(string prefix, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.GetOrCreatePathPrefix(prefix, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task AddFileAsync(long prefixid, string filename, DateTime lastmodified, long blocksetid, long metadataid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AddFile(prefixid, filename, lastmodified, blocksetid, metadataid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task AddUnmodifiedAsync(long fileid, DateTime lastModified, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AddKnownFile(fileid, lastModified, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
|
|
public Task MoveBlockToVolumeAsync(string blockkey, long size, long sourcevolumeid, long targetvolumeid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.MoveBlockToVolume(blockkey, size, sourcevolumeid, targetvolumeid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task SafeDeleteRemoteVolumeAsync(string remotename, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.SafeDeleteRemoteVolume(remotename, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<string[]> GetBlocklistHashesAsync(string remotename, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.GetBlocklistHashes(remotename, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task UpdateIndexVolumeAsync(IndexVolumeWriter indexvolume, BlockVolumeWriter blockvolume, CancellationToken cancellationToken)
|
|
{
|
|
if (indexvolume != null)
|
|
return Task.FromResult<bool>(true);
|
|
|
|
return RunOnMain(async () =>
|
|
{
|
|
await m_database
|
|
.AddIndexBlockLink(indexvolume.VolumeID, blockvolume.VolumeID, cancellationToken)
|
|
.ConfigureAwait(false);
|
|
|
|
indexvolume.StartVolume(blockvolume.RemoteFilename);
|
|
|
|
await foreach (var b in m_database.GetBlocks(blockvolume.VolumeID, cancellationToken).ConfigureAwait(false))
|
|
indexvolume.AddBlock(b.Hash, b.Size);
|
|
|
|
await m_database
|
|
.UpdateRemoteVolume(indexvolume.RemoteFilename, RemoteVolumeState.Uploading, -1, null, cancellationToken)
|
|
.ConfigureAwait(false);
|
|
});
|
|
}
|
|
|
|
public Task AppendFilesFromPreviousSetAsync(CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AppendFilesFromPreviousSet(null, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task AppendFilesFromPreviousSetAsync(string[] deletedFilelist, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AppendFilesFromPreviousSet(deletedFilelist, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task AppendFilesFromPreviousSetAsync(string[] deletedFilelist, long filesetid, long prevId, DateTime timestamp, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AppendFilesFromPreviousSet(deletedFilelist, filesetid, prevId, timestamp, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Populates FilesetEntry table with files from previous fileset, which aren't
|
|
/// yet part of the new fileset, and which aren't excluded by the (optional) exclusion
|
|
/// predicate.
|
|
/// </summary>
|
|
/// <param name="exclusionPredicate">Optional exclusion predicate (true = exclude file)</param>
|
|
public Task AppendFilesFromPreviousSetWithPredicateAsync(Func<string, long, bool> exclusionPredicate, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AppendFilesFromPreviousSetWithPredicate(exclusionPredicate, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Populates FilesetEntry table with files from previous fileset, which aren't
|
|
/// yet part of the new fileset, and which aren't excluded by the (optional) exclusion
|
|
/// predicate.
|
|
/// </summary>
|
|
/// <param name="exclusionPredicate">Optional exclusion predicate (true = exclude file).</param>
|
|
/// <param name="fileSetId">Current fileset ID.</param>
|
|
/// <param name="prevFileSetId">Source fileset ID.</param>
|
|
/// <param name="timestamp">If <c>prevFileSetId</c> == -1, used to locate previous fileset.</param>
|
|
/// <param name="cancellationToken">The cancellation token to cancel the operation.</param>
|
|
/// <returns>Task that completes when the files have been appended.</returns>
|
|
public Task AppendFilesFromPreviousSetWithPredicateAsync(Func<string, long, bool> exclusionPredicate, long fileSetId, long prevFileSetId, DateTime timestamp, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.AppendFilesFromPreviousSetWithPredicate(exclusionPredicate, fileSetId, prevFileSetId, timestamp, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<KeyValuePair<long, DateTime>[]> GetIncompleteFilesetsAsync(CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.GetIncompleteFilesets(cancellationToken)
|
|
.OrderBy(x => x.Value)
|
|
.ToArrayAsync(cancellationToken: cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<KeyValuePair<long, DateTime>[]> GetFilesetTimesAsync(CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.FilesetTimes(cancellationToken)
|
|
.ToArrayAsync(cancellationToken: cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<long> CreateFilesetAsync(long volumeID, DateTime fileTime, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
{
|
|
var fs = await m_database
|
|
.CreateFileset(volumeID, fileTime, cancellationToken)
|
|
.ConfigureAwait(false);
|
|
|
|
await m_database.Transaction
|
|
.CommitAsync(cancellationToken)
|
|
.ConfigureAwait(false);
|
|
|
|
return fs;
|
|
});
|
|
}
|
|
|
|
public Task LinkFilesetToVolumeAsync(long filesetid, long volumeid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.LinkFilesetToVolume(filesetid, volumeid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task WriteFilesetAsync(FilesetVolumeWriter fsw, long filesetid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.WriteFileset(fsw, filesetid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task PushTimestampChangesToPreviousVersionAsync(long filesetid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.PushTimestampChangesToPreviousVersion(filesetid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task UpdateFilesetAndMarkAsFullBackupAsync(long filesetid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.UpdateFullBackupStateInFileset(filesetid, true, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<IAsyncEnumerable<string>> GetMissingIndexFilesAsync(CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(() => m_database.GetMissingIndexFiles(cancellationToken));
|
|
}
|
|
|
|
public Task UpdateChangeStatisticsAsync(BackupResults result, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.UpdateChangeStatistics(result, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task VerifyConsistencyAsync(int blocksize, int blockhashSize, bool verifyfilelists, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.VerifyConsistency(blocksize, blockhashSize, verifyfilelists, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task RemoveRemoteVolumeAsync(string remoteFilename, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.RemoveRemoteVolume(remoteFilename, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<RemoteVolumeEntry> GetRemoteVolumeFromFilesetIDAsync(long filesetID, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.GetRemoteVolumeFromFilesetID(filesetID, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task CreateChangeJournalDataAsync(IEnumerable<USNJournalDataEntry> journalData, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.CreateChangeJournalData(journalData, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task UpdateChangeJournalDataAsync(IEnumerable<USNJournalDataEntry> journalData, long lastfilesetid, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.UpdateChangeJournalData(journalData, lastfilesetid, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
public Task<bool> IsBlocklistHashKnownAsync(string hash, CancellationToken cancellationToken)
|
|
{
|
|
return RunOnMain(async () =>
|
|
await m_database
|
|
.IsBlocklistHashKnown(hash, cancellationToken)
|
|
.ConfigureAwait(false)
|
|
);
|
|
}
|
|
|
|
}
|
|
}
|
|
|