1
0
Fork 0
mirror of https://github.com/transmission/transmission synced 2025-01-03 05:25:52 +00:00
transmission/libtransmission/torrent-files.h
Lucas Clemente Vella 9fc9daf40d
Fails with an error if data removal was not possible (#6055)
* Do not remove torrent if trashing files failed.

Instead, stop the torrent and sets an error.

* Fixing GTK interface with new file removal behavior.

* C++17 compliant.

* Reverting unrelated change.

* Avoiding allocating unecessary objects.

* Easy review fixes.

* Fixing merge error.

* Adding result callback to tr_torrentRemove().

Using the new callback in Gtk GUI to decide when to remove it from the
interface.

* Reducing indentation level and making the function more readable.

* Using existing Session RefPtr.

* Notifying the client before freeing the torrent in the session.

* Addressing comments and synchronizing callback.

* Moving include.

* Fix constness issue reported by clang-tidy
2024-08-13 05:26:09 +01:00

200 lines
5.3 KiB
C++

// This file Copyright © Mnemosyne LLC.
// It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
// or any future license endorsed by Mnemosyne LLC.
// License text can be found in the licenses/ folder.
#pragma once
#include <algorithm> // std::sort()
#include <cstddef>
#include <cstdint> // uint64_t
#include <functional>
#include <iterator>
#include <optional>
#include <string>
#include <string_view>
#include <utility>
#include <vector>
#include "libtransmission/transmission.h"
#include "libtransmission/file.h"
#include "libtransmission/tr-macros.h"
#include "libtransmission/tr-strbuf.h"
struct tr_error;
/**
* A simple collection of files & utils for finding them, moving them, etc.
*/
struct tr_torrent_files
{
public:
[[nodiscard]] TR_CONSTEXPR20 bool empty() const noexcept
{
return std::empty(files_);
}
[[nodiscard]] TR_CONSTEXPR20 size_t file_count() const noexcept
{
return std::size(files_);
}
[[nodiscard]] TR_CONSTEXPR20 uint64_t file_size(tr_file_index_t file_index) const
{
return files_.at(file_index).size_;
}
[[nodiscard]] constexpr auto total_size() const noexcept
{
return total_size_;
}
[[nodiscard]] TR_CONSTEXPR20 std::string const& path(tr_file_index_t file_index) const
{
return files_.at(file_index).path_;
}
void set_path(tr_file_index_t file_index, std::string_view path)
{
files_.at(file_index).set_path(path);
}
void insert_subpath_prefix(std::string_view path)
{
auto const buf = tr_pathbuf{ path, '/' };
for (auto& file : files_)
{
file.path_.insert(0, buf.sv());
file.path_.shrink_to_fit();
}
}
void reserve(size_t n_files)
{
files_.reserve(n_files);
}
void shrink_to_fit()
{
files_.shrink_to_fit();
}
TR_CONSTEXPR20 void clear() noexcept
{
files_.clear();
total_size_ = uint64_t{};
}
[[nodiscard]] auto sorted_by_path() const
{
auto ret = std::vector<std::pair<std::string /*path*/, uint64_t /*size*/>>{};
ret.reserve(std::size(files_));
std::transform(
std::begin(files_),
std::end(files_),
std::back_inserter(ret),
[](auto const& in) { return std::make_pair(in.path_, in.size_); });
std::sort(std::begin(ret), std::end(ret), [](auto const& lhs, auto const& rhs) { return lhs.first < rhs.first; });
return ret;
}
tr_file_index_t add(std::string_view path, uint64_t file_size)
{
auto const ret = static_cast<tr_file_index_t>(std::size(files_));
files_.emplace_back(path, file_size);
total_size_ += file_size;
return ret;
}
bool move(
std::string_view old_parent_in,
std::string_view parent_in,
std::string_view parent_name = "",
tr_error* error = nullptr) const;
using FileFunc = std::function<void(char const* filename)>;
void remove(std::string_view parent_in, std::string_view tmpdir_prefix, FileFunc const& func, tr_error* error = nullptr)
const;
struct FoundFile : public tr_sys_path_info
{
public:
FoundFile(tr_sys_path_info info, tr_pathbuf&& filename_in, size_t base_len_in)
: tr_sys_path_info{ info }
, filename_{ std::move(filename_in) }
, base_len_{ base_len_in }
{
}
[[nodiscard]] constexpr auto const& filename() const noexcept
{
// /home/foo/Downloads/torrent/01-file-one.txt
return filename_;
}
[[nodiscard]] constexpr auto base() const noexcept
{
// /home/foo/Downloads
return filename_.sv().substr(0, base_len_);
}
[[nodiscard]] constexpr auto subpath() const noexcept
{
// torrent/01-file-one.txt
return filename_.sv().substr(base_len_ + 1);
}
private:
tr_pathbuf filename_;
size_t base_len_;
};
[[nodiscard]] std::optional<FoundFile> find(tr_file_index_t file, std::string_view const* paths, size_t n_paths) const;
[[nodiscard]] bool has_any_local_data(std::string_view const* paths, size_t n_paths) const;
static void sanitize_subpath(std::string_view path, tr_pathbuf& append_me, bool os_specific = true);
[[nodiscard]] static auto sanitize_subpath(std::string_view path, bool os_specific = true)
{
auto tmp = tr_pathbuf{};
sanitize_subpath(path, tmp, os_specific);
return std::string{ tmp.sv() };
}
[[nodiscard]] static bool is_subpath_sanitized(std::string_view path, bool os_specific = true)
{
return sanitize_subpath(path, os_specific) == path;
}
static constexpr std::string_view PartialFileSuffix = ".part";
private:
struct file_t
{
public:
void set_path(std::string_view subpath)
{
if (path_ != subpath)
{
path_ = subpath;
path_.shrink_to_fit();
}
}
file_t(std::string_view path, uint64_t size)
: path_{ path }
, size_{ size }
{
}
std::string path_;
uint64_t size_ = 0;
};
std::vector<file_t> files_;
uint64_t total_size_ = 0;
};