mirror of https://github.com/procxx/kepka.git
Work with Cache::Database in LocalStorageBox.
This commit is contained in:
parent
55f60866cb
commit
08ff324b1b
|
@ -355,14 +355,21 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
"lng_settings_section_privacy" = "Privacy and Security";
|
||||
|
||||
"lng_local_storage_title" = "Local storage";
|
||||
"lng_settings_no_data_cached" = "No cached data found!";
|
||||
"lng_settings_images_cached#one" = "{count} image, {size}";
|
||||
"lng_settings_images_cached#other" = "{count} images, {size}";
|
||||
"lng_settings_audios_cached#one" = "{count} voice message, {size}";
|
||||
"lng_settings_audios_cached#other" = "{count} voice messages, {size}";
|
||||
"lng_local_storage_empty" = "No cached files";
|
||||
"lng_local_storage_image#one" = "{count} image";
|
||||
"lng_local_storage_image#other" = "{count} images";
|
||||
"lng_local_storage_sticker#one" = "{count} sticker";
|
||||
"lng_local_storage_sticker#other" = "{count} stickers";
|
||||
"lng_local_storage_voice#one" = "{count} voice message";
|
||||
"lng_local_storage_voice#other" = "{count} voice messages";
|
||||
"lng_local_storage_round#one" = "{count} video message";
|
||||
"lng_local_storage_round#other" = "{count} video messages";
|
||||
"lng_local_storage_animation#one" = "{count} animation";
|
||||
"lng_local_storage_animation#other" = "{count} animations";
|
||||
"lng_local_storage_summary" = "Summary";
|
||||
"lng_local_storage_clear_some" = "Clear";
|
||||
"lng_local_storage_clear" = "Clear all";
|
||||
"lng_local_storage_clearing" = "Clearing...";
|
||||
"lng_local_storage_cleared" = "Cleared!";
|
||||
|
||||
"lng_settings_section_advanced_settings" = "Advanced Settings";
|
||||
"lng_settings_enable_night_theme" = "Enable night mode";
|
||||
|
|
|
@ -307,6 +307,15 @@ public:
|
|||
|
||||
};
|
||||
|
||||
flat_multi_map() = default;
|
||||
flat_multi_map(const flat_multi_map &other) = default;
|
||||
flat_multi_map(flat_multi_map &&other) = default;
|
||||
flat_multi_map &operator=(const flat_multi_map &other) {
|
||||
auto copy = other;
|
||||
return (*this = std::move(copy));
|
||||
}
|
||||
flat_multi_map &operator=(flat_multi_map &&other) = default;
|
||||
|
||||
size_type size() const {
|
||||
return impl().size();
|
||||
}
|
||||
|
|
|
@ -46,6 +46,27 @@ TEST_CASE("flat_maps should keep items sorted by key", "[flat_map]") {
|
|||
}
|
||||
}
|
||||
|
||||
TEST_CASE("simple flat_maps tests", "[flat_map]") {
|
||||
SECTION("copy constructor") {
|
||||
base::flat_map<int, string> v;
|
||||
v.emplace(0, "a");
|
||||
v.emplace(2, "b");
|
||||
auto u = v;
|
||||
REQUIRE(u.size() == 2);
|
||||
REQUIRE(u.find(0) == u.begin());
|
||||
REQUIRE(u.find(2) == u.end() - 1);
|
||||
}
|
||||
SECTION("assignment") {
|
||||
base::flat_map<int, string> v, u;
|
||||
v.emplace(0, "a");
|
||||
v.emplace(2, "b");
|
||||
u = v;
|
||||
REQUIRE(u.size() == 2);
|
||||
REQUIRE(u.find(0) == u.begin());
|
||||
REQUIRE(u.find(2) == u.end() - 1);
|
||||
}
|
||||
}
|
||||
|
||||
TEST_CASE("flat_maps custom comparator", "[flat_map]") {
|
||||
base::flat_map<int_wrap, string, int_wrap_comparator> v;
|
||||
v.emplace({ 0 }, "a");
|
||||
|
|
|
@ -354,7 +354,27 @@ peerListBox: PeerList(defaultPeerList) {
|
|||
}
|
||||
}
|
||||
|
||||
localStorageBoxSkip: 10px;
|
||||
localStorageRowHeight: 50px;
|
||||
localStorageRowPadding: margins(23px, 5px, 23px, 5px);
|
||||
localStorageRowTitle: FlatLabel(defaultFlatLabel) {
|
||||
textFg: windowBoldFg;
|
||||
maxHeight: 20px;
|
||||
style: TextStyle(defaultTextStyle) {
|
||||
font: font(14px semibold);
|
||||
linkFont: font(14px semibold);
|
||||
linkFontOver: font(14px semibold);
|
||||
}
|
||||
}
|
||||
localStorageRowSize: FlatLabel(defaultFlatLabel) {
|
||||
textFg: contactsStatusFg;
|
||||
maxHeight: 20px;
|
||||
style: TextStyle(defaultTextStyle) {
|
||||
font: font(14px);
|
||||
linkFont: font(14px);
|
||||
linkFontOver: font(14px);
|
||||
}
|
||||
}
|
||||
localStorageClear: defaultBoxButton;
|
||||
|
||||
shareRowsTop: 12px;
|
||||
shareRowHeight: 108px;
|
||||
|
|
|
@ -8,15 +8,185 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "boxes/local_storage_box.h"
|
||||
|
||||
#include "styles/style_boxes.h"
|
||||
#include "ui/wrap/vertical_layout.h"
|
||||
#include "ui/wrap/slide_wrap.h"
|
||||
#include "ui/widgets/labels.h"
|
||||
#include "ui/widgets/buttons.h"
|
||||
#include "ui/effects/radial_animation.h"
|
||||
#include "ui/widgets/shadow.h"
|
||||
#include "storage/localstorage.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "mainwindow.h"
|
||||
#include "auth_session.h"
|
||||
#include "layout.h"
|
||||
|
||||
LocalStorageBox::LocalStorageBox(QWidget *parent)
|
||||
: _clear(this, lang(lng_local_storage_clear), st::boxLinkButton) {
|
||||
class LocalStorageBox::Row : public Ui::RpWidget {
|
||||
public:
|
||||
Row(
|
||||
QWidget *parent,
|
||||
Fn<QString(size_type)> title,
|
||||
Fn<QString()> clear,
|
||||
const Database::TaggedSummary &data);
|
||||
|
||||
void update(const Database::TaggedSummary &data);
|
||||
void toggleProgress(bool shown);
|
||||
|
||||
rpl::producer<> clearRequests() const;
|
||||
|
||||
protected:
|
||||
int resizeGetHeight(int newWidth) override;
|
||||
void paintEvent(QPaintEvent *e) override;
|
||||
|
||||
private:
|
||||
QString titleText(const Database::TaggedSummary &data) const;
|
||||
QString sizeText(const Database::TaggedSummary &data) const;
|
||||
void step_radial(TimeMs ms, bool timer);
|
||||
|
||||
Fn<QString(size_type)> _titleFactory;
|
||||
object_ptr<Ui::FlatLabel> _title;
|
||||
object_ptr<Ui::FlatLabel> _description;
|
||||
object_ptr<Ui::FlatLabel> _clearing = { nullptr };
|
||||
object_ptr<Ui::RoundButton> _clear;
|
||||
std::unique_ptr<Ui::InfiniteRadialAnimation> _progress;
|
||||
|
||||
};
|
||||
|
||||
LocalStorageBox::Row::Row(
|
||||
QWidget *parent,
|
||||
Fn<QString(size_type)> title,
|
||||
Fn<QString()> clear,
|
||||
const Database::TaggedSummary &data)
|
||||
: RpWidget(parent)
|
||||
, _titleFactory(std::move(title))
|
||||
, _title(
|
||||
this,
|
||||
titleText(data),
|
||||
Ui::FlatLabel::InitType::Simple,
|
||||
st::localStorageRowTitle)
|
||||
, _description(
|
||||
this,
|
||||
sizeText(data),
|
||||
Ui::FlatLabel::InitType::Simple,
|
||||
st::localStorageRowSize)
|
||||
, _clear(this, std::move(clear), st::localStorageClear) {
|
||||
_clear->setVisible(data.count != 0);
|
||||
}
|
||||
|
||||
void LocalStorageBox::Row::update(const Database::TaggedSummary &data) {
|
||||
if (data.count != 0) {
|
||||
_title->setText(titleText(data));
|
||||
}
|
||||
_description->setText(sizeText(data));
|
||||
_clear->setVisible(data.count != 0);
|
||||
}
|
||||
|
||||
void LocalStorageBox::Row::toggleProgress(bool shown) {
|
||||
if (!shown) {
|
||||
_progress = nullptr;
|
||||
_description->show();
|
||||
_clearing.destroy();
|
||||
} else if (!_progress) {
|
||||
_progress = std::make_unique<Ui::InfiniteRadialAnimation>(
|
||||
animation(this, &Row::step_radial),
|
||||
st::proxyCheckingAnimation);
|
||||
_progress->start();
|
||||
_clearing = object_ptr<Ui::FlatLabel>(
|
||||
this,
|
||||
lang(lng_local_storage_clearing),
|
||||
Ui::FlatLabel::InitType::Simple,
|
||||
st::localStorageRowSize);
|
||||
_clearing->show();
|
||||
_description->hide();
|
||||
resizeToWidth(width());
|
||||
RpWidget::update();
|
||||
}
|
||||
}
|
||||
|
||||
void LocalStorageBox::Row::step_radial(TimeMs ms, bool timer) {
|
||||
if (timer) {
|
||||
RpWidget::update();
|
||||
}
|
||||
}
|
||||
|
||||
rpl::producer<> LocalStorageBox::Row::clearRequests() const {
|
||||
return _clear->clicks();
|
||||
}
|
||||
|
||||
int LocalStorageBox::Row::resizeGetHeight(int newWidth) {
|
||||
const auto height = st::localStorageRowHeight;
|
||||
const auto padding = st::localStorageRowPadding;
|
||||
const auto available = newWidth - padding.left() - padding.right();
|
||||
_title->resizeToWidth(available);
|
||||
_description->resizeToWidth(available);
|
||||
_title->moveToLeft(padding.left(), padding.top(), newWidth);
|
||||
_description->moveToLeft(
|
||||
padding.left(),
|
||||
height - padding.bottom() - _description->height(),
|
||||
newWidth);
|
||||
if (_clearing) {
|
||||
const auto progressShift = st::proxyCheckingPosition.x()
|
||||
+ st::proxyCheckingAnimation.size.width()
|
||||
+ st::proxyCheckingSkip;
|
||||
_clearing->resizeToWidth(available - progressShift);
|
||||
_clearing->moveToLeft(
|
||||
padding.left(),// + progressShift,
|
||||
_description->y(),
|
||||
newWidth);
|
||||
}
|
||||
_clear->moveToRight(
|
||||
st::boxButtonPadding.right(),
|
||||
(height - _clear->height()) / 2,
|
||||
newWidth);
|
||||
return height;
|
||||
}
|
||||
|
||||
void LocalStorageBox::Row::paintEvent(QPaintEvent *e) {
|
||||
if (!_progress || true) {
|
||||
return;
|
||||
}
|
||||
Painter p(this);
|
||||
|
||||
const auto padding = st::localStorageRowPadding;
|
||||
const auto height = st::localStorageRowHeight;
|
||||
const auto bottom = height - padding.bottom() - _description->height();
|
||||
_progress->step(crl::time());
|
||||
_progress->draw(
|
||||
p,
|
||||
{
|
||||
st::proxyCheckingPosition.x() + padding.left(),
|
||||
st::proxyCheckingPosition.y() + bottom
|
||||
},
|
||||
width());
|
||||
}
|
||||
|
||||
QString LocalStorageBox::Row::titleText(const Database::TaggedSummary &data) const {
|
||||
return _titleFactory(data.count);
|
||||
}
|
||||
|
||||
QString LocalStorageBox::Row::sizeText(const Database::TaggedSummary &data) const {
|
||||
return data.totalSize
|
||||
? formatSizeText(data.totalSize)
|
||||
: lang(lng_local_storage_empty);
|
||||
}
|
||||
|
||||
LocalStorageBox::LocalStorageBox(
|
||||
QWidget*,
|
||||
not_null<Database*> db,
|
||||
CreateTag)
|
||||
: _db(db) {
|
||||
}
|
||||
|
||||
void LocalStorageBox::Show(not_null<Database*> db) {
|
||||
auto shared = std::make_shared<object_ptr<LocalStorageBox>>(
|
||||
Box<LocalStorageBox>(db, CreateTag()));
|
||||
const auto weak = shared->data();
|
||||
db->statsOnMain(
|
||||
) | rpl::start_with_next([=](Database::Stats &&stats) {
|
||||
weak->update(std::move(stats));
|
||||
if (auto &strong = *shared) {
|
||||
Ui::show(std::move(strong));
|
||||
}
|
||||
}, weak->lifetime());
|
||||
}
|
||||
|
||||
void LocalStorageBox::prepare() {
|
||||
|
@ -24,27 +194,110 @@ void LocalStorageBox::prepare() {
|
|||
|
||||
addButton(langFactory(lng_box_ok), [this] { closeBox(); });
|
||||
|
||||
_clear->setClickedCallback([this] { clearStorage(); });
|
||||
|
||||
connect(App::wnd(), SIGNAL(tempDirCleared(int)), this, SLOT(onTempDirCleared(int)));
|
||||
connect(App::wnd(), SIGNAL(tempDirClearFailed(int)), this, SLOT(onTempDirClearFailed(int)));
|
||||
|
||||
subscribe(Auth().downloaderTaskFinished(), [this] { update(); });
|
||||
|
||||
updateControls();
|
||||
|
||||
checkLocalStoredCounts();
|
||||
setupControls();
|
||||
}
|
||||
|
||||
void LocalStorageBox::updateControls() {
|
||||
const auto rowsHeight = st::linkFont->height + st::localStorageBoxSkip;
|
||||
_clear->setVisible(false);
|
||||
setDimensions(st::boxWidth, st::localStorageBoxSkip + rowsHeight + _clear->height());
|
||||
_clear->moveToLeft(st::boxPadding.left(), st::localStorageBoxSkip + rowsHeight);
|
||||
update();
|
||||
void LocalStorageBox::updateRow(
|
||||
not_null<Ui::SlideWrap<Row>*> row,
|
||||
Database::TaggedSummary *data) {
|
||||
const auto summary = (_rows.find(0)->second == row);
|
||||
const auto shown = (data && data->count && data->totalSize) || summary;
|
||||
if (shown) {
|
||||
row->entity()->update(*data);
|
||||
}
|
||||
row->toggle(shown, anim::type::normal);
|
||||
}
|
||||
|
||||
void LocalStorageBox::checkLocalStoredCounts() {
|
||||
void LocalStorageBox::update(Database::Stats &&stats) {
|
||||
_stats = std::move(stats);
|
||||
if (const auto i = _rows.find(0); i != end(_rows)) {
|
||||
i->second->entity()->toggleProgress(_stats.clearing);
|
||||
}
|
||||
for (const auto &entry : _rows) {
|
||||
if (entry.first) {
|
||||
const auto i = _stats.tagged.find(entry.first);
|
||||
updateRow(
|
||||
entry.second,
|
||||
(i != end(_stats.tagged)) ? &i->second : nullptr);
|
||||
} else {
|
||||
updateRow(entry.second, &_stats.full);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void LocalStorageBox::clearByTag(uint8 tag) {
|
||||
if (tag) {
|
||||
_db->clearByTag(tag);
|
||||
} else {
|
||||
_db->clear();
|
||||
}
|
||||
}
|
||||
|
||||
void LocalStorageBox::setupControls() {
|
||||
_content.create(this);
|
||||
|
||||
const auto createRow = [&](
|
||||
uint8 tag,
|
||||
Fn<QString(size_type)> title,
|
||||
Fn<QString()> clear,
|
||||
const Database::TaggedSummary &data) {
|
||||
auto result = _content->add(object_ptr<Ui::SlideWrap<Row>>(
|
||||
_content,
|
||||
object_ptr<Row>(
|
||||
_content,
|
||||
std::move(title),
|
||||
std::move(clear),
|
||||
data)));
|
||||
const auto shown = (data.count && data.totalSize) || !tag;
|
||||
result->toggle(shown, anim::type::instant);
|
||||
result->entity()->clearRequests(
|
||||
) | rpl::start_with_next([=] {
|
||||
clearByTag(tag);
|
||||
}, result->lifetime());
|
||||
_rows.emplace(tag, result);
|
||||
return result;
|
||||
};
|
||||
auto tracker = Ui::MultiSlideTracker();
|
||||
const auto createTagRow = [&](uint8 tag, auto &&titleFactory) {
|
||||
static const auto empty = Database::TaggedSummary();
|
||||
const auto i = _stats.tagged.find(tag);
|
||||
const auto &data = (i != end(_stats.tagged)) ? i->second : empty;
|
||||
auto factory = std::forward<decltype(titleFactory)>(titleFactory);
|
||||
auto title = [factory = std::move(factory)](size_type count) {
|
||||
return factory(lt_count, count);
|
||||
};
|
||||
tracker.track(createRow(
|
||||
tag,
|
||||
std::move(title),
|
||||
langFactory(lng_local_storage_clear_some),
|
||||
data));
|
||||
};
|
||||
auto summaryTitle = [](size_type) {
|
||||
return lang(lng_local_storage_summary);
|
||||
};
|
||||
createRow(
|
||||
0,
|
||||
std::move(summaryTitle),
|
||||
langFactory(lng_local_storage_clear),
|
||||
_stats.full);
|
||||
const auto shadow = _content->add(object_ptr<Ui::SlideWrap<>>(
|
||||
_content,
|
||||
object_ptr<Ui::PlainShadow>(_content),
|
||||
st::localStorageRowPadding)
|
||||
);
|
||||
createTagRow(Data::kImageCacheTag, lng_local_storage_image);
|
||||
createTagRow(Data::kStickerCacheTag, lng_local_storage_sticker);
|
||||
createTagRow(Data::kVoiceMessageCacheTag, lng_local_storage_voice);
|
||||
createTagRow(Data::kVideoMessageCacheTag, lng_local_storage_round);
|
||||
createTagRow(Data::kAnimationCacheTag, lng_local_storage_animation);
|
||||
shadow->toggleOn(
|
||||
std::move(tracker).atLeastOneShownValue()
|
||||
);
|
||||
_content->resizeToWidth(st::boxWidth);
|
||||
_content->heightValue(
|
||||
) | rpl::start_with_next([=](int height) {
|
||||
setDimensions(st::boxWidth, height);
|
||||
}, _content->lifetime());
|
||||
}
|
||||
|
||||
void LocalStorageBox::paintEvent(QPaintEvent *e) {
|
||||
|
@ -54,40 +307,4 @@ void LocalStorageBox::paintEvent(QPaintEvent *e) {
|
|||
|
||||
p.setFont(st::boxTextFont);
|
||||
p.setPen(st::windowFg);
|
||||
checkLocalStoredCounts();
|
||||
auto top = st::localStorageBoxSkip;
|
||||
p.drawTextLeft(st::boxPadding.left(), top, width(), lang(lng_settings_no_data_cached));
|
||||
top += st::boxTextFont->height + st::localStorageBoxSkip;
|
||||
auto text = ([this]() -> QString {
|
||||
switch (_state) {
|
||||
case State::Clearing: return lang(lng_local_storage_clearing);
|
||||
case State::Cleared: return lang(lng_local_storage_cleared);
|
||||
case State::ClearFailed: return Lang::Hard::ClearPathFailed();
|
||||
}
|
||||
return QString();
|
||||
})();
|
||||
if (!text.isEmpty()) {
|
||||
p.drawTextLeft(st::boxPadding.left(), top, width(), text);
|
||||
top += st::boxTextFont->height + st::localStorageBoxSkip;
|
||||
}
|
||||
}
|
||||
|
||||
void LocalStorageBox::clearStorage() {
|
||||
App::wnd()->tempDirDelete(Local::ClearManagerStorage);
|
||||
_state = State::Clearing;
|
||||
updateControls();
|
||||
}
|
||||
|
||||
void LocalStorageBox::onTempDirCleared(int task) {
|
||||
if (task & Local::ClearManagerStorage) {
|
||||
_state = State::Cleared;
|
||||
}
|
||||
updateControls();
|
||||
}
|
||||
|
||||
void LocalStorageBox::onTempDirClearFailed(int task) {
|
||||
if (task & Local::ClearManagerStorage) {
|
||||
_state = State::ClearFailed;
|
||||
}
|
||||
updateControls();
|
||||
}
|
||||
|
|
|
@ -8,20 +8,30 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#pragma once
|
||||
|
||||
#include "boxes/abstract_box.h"
|
||||
#include "storage/cache/storage_cache_database.h"
|
||||
|
||||
namespace Storage {
|
||||
namespace Cache {
|
||||
class Database;
|
||||
} // namespace Cache
|
||||
} // namespace Storage
|
||||
|
||||
namespace Ui {
|
||||
class LinkButton;
|
||||
class VerticalLayout;
|
||||
template <typename Widget>
|
||||
class SlideWrap;
|
||||
} // namespace Ui
|
||||
|
||||
class LocalStorageBox : public BoxContent {
|
||||
Q_OBJECT
|
||||
struct CreateTag {
|
||||
};
|
||||
|
||||
public:
|
||||
LocalStorageBox(QWidget*);
|
||||
using Database = Storage::Cache::Database;
|
||||
|
||||
private slots:
|
||||
void onTempDirCleared(int task);
|
||||
void onTempDirClearFailed(int task);
|
||||
LocalStorageBox(QWidget*, not_null<Database*> db, CreateTag);
|
||||
|
||||
static void Show(not_null<Database*> db);
|
||||
|
||||
protected:
|
||||
void prepare() override;
|
||||
|
@ -29,18 +39,19 @@ protected:
|
|||
void paintEvent(QPaintEvent *e) override;
|
||||
|
||||
private:
|
||||
void clearStorage();
|
||||
void updateControls();
|
||||
void checkLocalStoredCounts();
|
||||
class Row;
|
||||
|
||||
enum class State {
|
||||
Normal,
|
||||
Clearing,
|
||||
Cleared,
|
||||
ClearFailed,
|
||||
};
|
||||
State _state = State::Normal;
|
||||
void clearByTag(uint8 tag);
|
||||
void update(Database::Stats &&stats);
|
||||
void updateRow(
|
||||
not_null<Ui::SlideWrap<Row>*> row,
|
||||
Database::TaggedSummary *data);
|
||||
void setupControls();
|
||||
|
||||
object_ptr<Ui::LinkButton> _clear;
|
||||
not_null<Storage::Cache::Database*> _db;
|
||||
Database::Stats _stats;
|
||||
|
||||
object_ptr<Ui::VerticalLayout> _content = { nullptr };
|
||||
base::flat_map<uint8, not_null<Ui::SlideWrap<Row>*>> _rows;
|
||||
|
||||
};
|
||||
|
|
|
@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "observer_peer.h"
|
||||
#include "auth_session.h"
|
||||
#include "apiwrap.h"
|
||||
#include "messenger.h"
|
||||
#include "export/export_controller.h"
|
||||
#include "export/view/export_view_panel_controller.h"
|
||||
#include "window/notifications_manager.h"
|
||||
|
@ -67,17 +68,19 @@ void UpdateImage(ImagePtr &old, ImagePtr now) {
|
|||
|
||||
Session::Session(not_null<AuthSession*> session)
|
||||
: _session(session)
|
||||
, _cache(Local::cachePath(), Local::cacheSettings())
|
||||
, _cache(Messenger::Instance().databases().get(
|
||||
Local::cachePath(),
|
||||
Local::cacheSettings()))
|
||||
, _groups(this)
|
||||
, _unmuteByFinishedTimer([=] { unmuteByFinished(); }) {
|
||||
_cache.open(Local::cacheKey());
|
||||
_cache->open(Local::cacheKey());
|
||||
|
||||
setupContactViewsViewer();
|
||||
setupChannelLeavingViewer();
|
||||
}
|
||||
|
||||
Storage::Cache::Database &Session::cache() {
|
||||
return _cache;
|
||||
return *_cache;
|
||||
}
|
||||
|
||||
void Session::startExport(PeerData *peer) {
|
||||
|
|
|
@ -7,7 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
*/
|
||||
#pragma once
|
||||
|
||||
#include "storage/cache/storage_cache_database.h"
|
||||
#include "storage/storage_databases.h"
|
||||
#include "chat_helpers/stickers.h"
|
||||
#include "dialogs/dialogs_key.h"
|
||||
#include "data/data_groups.h"
|
||||
|
@ -523,7 +523,7 @@ private:
|
|||
|
||||
not_null<AuthSession*> _session;
|
||||
|
||||
Storage::Cache::Database _cache;
|
||||
Storage::DatabasePointer _cache;
|
||||
|
||||
std::unique_ptr<Export::ControllerWrap> _export;
|
||||
std::unique_ptr<Export::View::PanelController> _exportPanel;
|
||||
|
|
|
@ -30,6 +30,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "passport/passport_form_controller.h"
|
||||
#include "observer_peer.h"
|
||||
#include "storage/file_upload.h"
|
||||
#include "storage/storage_databases.h"
|
||||
#include "mainwidget.h"
|
||||
#include "mediaview.h"
|
||||
#include "mtproto/dc_options.h"
|
||||
|
@ -76,6 +77,7 @@ Messenger::Messenger(not_null<Core::Launcher*> launcher)
|
|||
: QObject()
|
||||
, _launcher(launcher)
|
||||
, _private(std::make_unique<Private>())
|
||||
, _databases(std::make_unique<Storage::Databases>())
|
||||
, _langpack(std::make_unique<Lang::Instance>())
|
||||
, _audio(std::make_unique<Media::Audio::Instance>())
|
||||
, _logo(Window::LoadLogo())
|
||||
|
|
|
@ -19,6 +19,10 @@ class Translator;
|
|||
class MediaView;
|
||||
class BoxContent;
|
||||
|
||||
namespace Storage {
|
||||
class Databases;
|
||||
} // namespace Storage
|
||||
|
||||
namespace Core {
|
||||
class Launcher;
|
||||
} // namespace Core
|
||||
|
@ -125,6 +129,11 @@ public:
|
|||
void suggestMainDcId(MTP::DcId mainDcId);
|
||||
void destroyStaleAuthorizationKeys();
|
||||
|
||||
// Databases
|
||||
Storage::Databases &databases() {
|
||||
return *_databases;
|
||||
}
|
||||
|
||||
// AuthSession component.
|
||||
AuthSession *authSession() {
|
||||
return _authSession.get();
|
||||
|
@ -249,6 +258,7 @@ private:
|
|||
|
||||
QWidget _globalShortcutParent;
|
||||
|
||||
std::unique_ptr<Storage::Databases> _databases;
|
||||
std::unique_ptr<MainWindow> _window;
|
||||
std::unique_ptr<MediaView> _mediaView;
|
||||
std::unique_ptr<Lang::Instance> _langpack;
|
||||
|
|
|
@ -34,7 +34,7 @@ public:
|
|||
void fire_copy(const Value &value) const {
|
||||
return fire_forward(value);
|
||||
}
|
||||
#if defined _MSC_VER && _MSC_VER >= 1914
|
||||
#if defined _MSC_VER && _MSC_VER >= 1914 && false
|
||||
producer<Value> events() const {
|
||||
#else // _MSC_VER >= 1914
|
||||
auto events() const {
|
||||
|
@ -65,6 +65,9 @@ public:
|
|||
auto events_starting_with_copy(const Value &value) const {
|
||||
return single(value) | then(events());
|
||||
}
|
||||
bool has_consumers() const {
|
||||
return (_data != nullptr) && !_data->consumers.empty();
|
||||
}
|
||||
|
||||
~event_stream();
|
||||
|
||||
|
|
|
@ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "boxes/confirm_box.h"
|
||||
#include "boxes/about_box.h"
|
||||
#include "boxes/local_storage_box.h"
|
||||
#include "data/data_session.h"
|
||||
#include "mainwindow.h"
|
||||
#include "ui/widgets/buttons.h"
|
||||
#include "ui/wrap/slide_wrap.h"
|
||||
|
@ -85,7 +86,7 @@ void AdvancedWidget::checkNonDefaultTheme() {
|
|||
}
|
||||
|
||||
void AdvancedWidget::onManageLocalStorage() {
|
||||
Ui::show(Box<LocalStorageBox>());
|
||||
LocalStorageBox::Show(&Auth().data().cache());
|
||||
}
|
||||
|
||||
#ifndef TDESKTOP_DISABLE_NETWORK_PROXY
|
||||
|
|
|
@ -16,6 +16,12 @@ Database::Database(const QString &path, const Settings &settings)
|
|||
: _wrapped(path, settings) {
|
||||
}
|
||||
|
||||
void Database::reconfigure(const Settings &settings) {
|
||||
_wrapped.with([settings](Implementation &unwrapped) mutable {
|
||||
unwrapped.reconfigure(settings);
|
||||
});
|
||||
}
|
||||
|
||||
void Database::open(EncryptionKey &&key, FnMut<void(Error)> &&done) {
|
||||
_wrapped.with([
|
||||
key = std::move(key),
|
||||
|
@ -125,7 +131,7 @@ void Database::putIfEmpty(
|
|||
void Database::getWithTag(
|
||||
const Key &key,
|
||||
FnMut<void(TaggedValue&&)> &&done) {
|
||||
_wrapped.with([
|
||||
_wrapped.with([
|
||||
key,
|
||||
done = std::move(done)
|
||||
](Implementation &unwrapped) mutable {
|
||||
|
@ -133,11 +139,9 @@ void Database::getWithTag(
|
|||
});
|
||||
}
|
||||
|
||||
void Database::stats(FnMut<void(Stats&&)> &&done) {
|
||||
_wrapped.with([
|
||||
done = std::move(done)
|
||||
](Implementation &unwrapped) mutable {
|
||||
unwrapped.stats(std::move(done));
|
||||
auto Database::statsOnMain() const -> rpl::producer<Stats> {
|
||||
return _wrapped.producer_on_main([](const Implementation &unwrapped) {
|
||||
return unwrapped.stats();
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -149,6 +153,15 @@ void Database::clear(FnMut<void(Error)> &&done) {
|
|||
});
|
||||
}
|
||||
|
||||
void Database::clearByTag(uint8 tag, FnMut<void(Error)> &&done) {
|
||||
_wrapped.with([
|
||||
tag,
|
||||
done = std::move(done)
|
||||
](Implementation &unwrapped) mutable {
|
||||
unwrapped.clearByTag(tag, std::move(done));
|
||||
});
|
||||
}
|
||||
|
||||
Database::~Database() = default;
|
||||
|
||||
} // namespace Cache
|
||||
|
|
|
@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "base/basic_types.h"
|
||||
#include <crl/crl_object_on_queue.h>
|
||||
#include <crl/crl_time.h>
|
||||
#include <rpl/producer.h>
|
||||
#include <QtCore/QString>
|
||||
|
||||
namespace Storage {
|
||||
|
@ -25,6 +26,8 @@ public:
|
|||
using Settings = details::Settings;
|
||||
Database(const QString &path, const Settings &settings);
|
||||
|
||||
void reconfigure(const Settings &settings);
|
||||
|
||||
void open(EncryptionKey &&key, FnMut<void(Error)> &&done = nullptr);
|
||||
void close(FnMut<void()> &&done = nullptr);
|
||||
|
||||
|
@ -60,9 +63,11 @@ public:
|
|||
void getWithTag(const Key &key, FnMut<void(TaggedValue&&)> &&done);
|
||||
|
||||
using Stats = details::Stats;
|
||||
void stats(FnMut<void(Stats&&)> &&done);
|
||||
using TaggedSummary = details::TaggedSummary;
|
||||
rpl::producer<Stats> statsOnMain() const;
|
||||
|
||||
void clear(FnMut<void(Error)> &&done = nullptr);
|
||||
void clearByTag(uint8 tag, FnMut<void(Error)> &&done = nullptr);
|
||||
|
||||
~Database();
|
||||
|
||||
|
|
|
@ -82,6 +82,18 @@ DatabaseObject::DatabaseObject(
|
|||
, _settings(settings)
|
||||
, _writeBundlesTimer(_weak, [=] { writeBundles(); checkCompactor(); })
|
||||
, _pruneTimer(_weak, [=] { prune(); }) {
|
||||
checkSettings();
|
||||
}
|
||||
|
||||
void DatabaseObject::reconfigure(const Settings &settings) {
|
||||
Expects(_key.empty());
|
||||
|
||||
_settings = settings;
|
||||
checkSettings();
|
||||
}
|
||||
|
||||
void DatabaseObject::checkSettings() {
|
||||
Expects(_settings.staleRemoveChunk > 0);
|
||||
Expects(_settings.maxDataSize > 0
|
||||
&& _settings.maxDataSize < kDataSizeLimit);
|
||||
Expects(_settings.maxBundledRecords > 0
|
||||
|
@ -124,7 +136,9 @@ Error DatabaseObject::openSomeBinlog(EncryptionKey &&key) {
|
|||
case File::Result::LockFailed:
|
||||
return Error{ Error::Type::LockFailed, binlogPath(version) };
|
||||
case File::Result::WrongKey:
|
||||
return Error{ Error::Type::WrongKey, binlogPath(version) };
|
||||
return _settings.clearOnWrongKey
|
||||
? openNewBinlog(key)
|
||||
: Error{ Error::Type::WrongKey, binlogPath(version) };
|
||||
}
|
||||
Unexpected("Result from DatabaseObject::openBinlog.");
|
||||
}
|
||||
|
@ -290,8 +304,8 @@ bool DatabaseObject::startDelayedPruning() {
|
|||
if (_settings.totalSizeLimit > 0
|
||||
&& _totalSize > _settings.totalSizeLimit) {
|
||||
return true;
|
||||
} else if (_minimalEntryTime != 0
|
||||
&& _minimalEntryTime <= before) {
|
||||
} else if ((!_minimalEntryTime && !_map.empty())
|
||||
|| _minimalEntryTime <= before) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
|
@ -315,17 +329,77 @@ bool DatabaseObject::startDelayedPruning() {
|
|||
}
|
||||
|
||||
void DatabaseObject::prune() {
|
||||
if (!_stale.empty()) {
|
||||
return;
|
||||
}
|
||||
auto stale = base::flat_set<Key>();
|
||||
auto staleTotalSize = int64();
|
||||
collectTimePrune(stale, staleTotalSize);
|
||||
collectSizePrune(stale, staleTotalSize);
|
||||
collectTimeStale(stale, staleTotalSize);
|
||||
collectSizeStale(stale, staleTotalSize);
|
||||
if (stale.size() <= _settings.staleRemoveChunk) {
|
||||
clearStaleNow(stale);
|
||||
} else {
|
||||
_stale = ranges::view::all(stale) | ranges::to_vector;
|
||||
startStaleClear();
|
||||
}
|
||||
}
|
||||
|
||||
void DatabaseObject::startStaleClear() {
|
||||
// Report "Clearing..." status.
|
||||
pushStats();
|
||||
clearStaleChunk();
|
||||
}
|
||||
|
||||
void DatabaseObject::clearStaleNow(const base::flat_set<Key> &stale) {
|
||||
if (stale.empty()) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Report "Clearing..." status.
|
||||
_stale.push_back(stale.front());
|
||||
pushStats();
|
||||
|
||||
for (const auto &key : stale) {
|
||||
remove(key, nullptr);
|
||||
}
|
||||
|
||||
// Report correct status async.
|
||||
_stale.clear();
|
||||
optimize();
|
||||
}
|
||||
|
||||
void DatabaseObject::collectTimePrune(
|
||||
void DatabaseObject::clearStaleChunkDelayed() {
|
||||
if (_clearingStale) {
|
||||
return;
|
||||
}
|
||||
_clearingStale = true;
|
||||
_weak.with([](DatabaseObject &that) {
|
||||
if (base::take(that._clearingStale)) {
|
||||
that.clearStaleChunk();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
void DatabaseObject::clearStaleChunk() {
|
||||
if (_stale.empty()) {
|
||||
return;
|
||||
}
|
||||
const auto stale = gsl::make_span(_stale);
|
||||
const auto count = size_type(stale.size());
|
||||
const auto clear = std::min(count, _settings.staleRemoveChunk);
|
||||
for (const auto &key : stale.subspan(count - clear)) {
|
||||
remove(key, nullptr);
|
||||
}
|
||||
_stale.resize(count - clear);
|
||||
if (_stale.empty()) {
|
||||
base::take(_stale);
|
||||
optimize();
|
||||
} else {
|
||||
clearStaleChunkDelayed();
|
||||
}
|
||||
}
|
||||
|
||||
void DatabaseObject::collectTimeStale(
|
||||
base::flat_set<Key> &stale,
|
||||
int64 &staleTotalSize) {
|
||||
if (!_settings.totalTimeLimit) {
|
||||
|
@ -351,7 +425,7 @@ void DatabaseObject::collectTimePrune(
|
|||
}
|
||||
}
|
||||
|
||||
void DatabaseObject::collectSizePrune(
|
||||
void DatabaseObject::collectSizeStale(
|
||||
base::flat_set<Key> &stale,
|
||||
int64 &staleTotalSize) {
|
||||
const auto removeSize = (_settings.totalSizeLimit > 0)
|
||||
|
@ -516,7 +590,9 @@ void DatabaseObject::setMapEntry(const Key &key, Entry &&entry) {
|
|||
++_entriesWithMinimalTimeCount;
|
||||
} else if (already.useTime == _minimalEntryTime) {
|
||||
Assert(_entriesWithMinimalTimeCount > 0);
|
||||
--_entriesWithMinimalTimeCount;
|
||||
if (!--_entriesWithMinimalTimeCount) {
|
||||
_minimalEntryTime = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
already = std::move(entry);
|
||||
|
@ -542,6 +618,25 @@ void DatabaseObject::updateStats(const Entry &was, const Entry &now) {
|
|||
summary.totalSize -= was.size;
|
||||
}
|
||||
}
|
||||
pushStatsDelayed();
|
||||
}
|
||||
|
||||
void DatabaseObject::pushStatsDelayed() {
|
||||
if (_pushingStats) {
|
||||
return;
|
||||
}
|
||||
_pushingStats = true;
|
||||
_weak.with([](DatabaseObject &that) {
|
||||
if (base::take(that._pushingStats)) {
|
||||
that.pushStats();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
void DatabaseObject::pushStats() {
|
||||
if (_stats.has_consumers()) {
|
||||
_stats.fire(collectStats());
|
||||
}
|
||||
}
|
||||
|
||||
void DatabaseObject::eraseMapEntry(const Map::const_iterator &i) {
|
||||
|
@ -550,7 +645,9 @@ void DatabaseObject::eraseMapEntry(const Map::const_iterator &i) {
|
|||
updateStats(entry, Entry());
|
||||
if (_minimalEntryTime != 0 && entry.useTime == _minimalEntryTime) {
|
||||
Assert(_entriesWithMinimalTimeCount > 0);
|
||||
--_entriesWithMinimalTimeCount;
|
||||
if (!--_entriesWithMinimalTimeCount) {
|
||||
_minimalEntryTime = 0;
|
||||
}
|
||||
}
|
||||
_map.erase(i);
|
||||
}
|
||||
|
@ -629,14 +726,29 @@ void DatabaseObject::compactorFail() {
|
|||
void DatabaseObject::close(FnMut<void()> &&done) {
|
||||
if (_binlog.isOpen()) {
|
||||
writeBundles();
|
||||
_binlog.close();
|
||||
}
|
||||
invokeCallback(done);
|
||||
clearState();
|
||||
}
|
||||
|
||||
void DatabaseObject::clearState() {
|
||||
_path = QString();
|
||||
_key = {};
|
||||
_map = {};
|
||||
_removing = {};
|
||||
_accessed = {};
|
||||
_time = {};
|
||||
_binlogExcessLength = 0;
|
||||
_totalSize = 0;
|
||||
_minimalEntryTime = 0;
|
||||
_entriesWithMinimalTimeCount = 0;
|
||||
_taggedStats = {};
|
||||
_pushingStats = false;
|
||||
_writeBundlesTimer.cancel();
|
||||
_pruneTimer.cancel();
|
||||
_cleaner = CleanerWrap();
|
||||
_compactor = CompactorWrap();
|
||||
_binlog.close();
|
||||
_key = EncryptionKey();
|
||||
invokeCallback(done);
|
||||
_map.clear();
|
||||
_binlogExcessLength = 0;
|
||||
}
|
||||
|
||||
void DatabaseObject::put(
|
||||
|
@ -648,6 +760,7 @@ void DatabaseObject::put(
|
|||
return;
|
||||
}
|
||||
_removing.erase(key);
|
||||
_stale.erase(ranges::remove(_stale, key), end(_stale));
|
||||
|
||||
const auto checksum = CountChecksum(bytes::make_span(value.bytes));
|
||||
const auto maybepath = writeKeyPlace(key, value, checksum);
|
||||
|
@ -762,6 +875,7 @@ Error DatabaseObject::writeExistingPlaceGeneric(
|
|||
const Key &key,
|
||||
const Entry &entry) {
|
||||
record.key = key;
|
||||
record.tag = entry.tag;
|
||||
record.size = ReadTo<EntrySize>(entry.size);
|
||||
record.checksum = entry.checksum;
|
||||
if (const auto i = _map.find(key); i != end(_map)) {
|
||||
|
@ -927,20 +1041,22 @@ void DatabaseObject::moveIfEmpty(
|
|||
invokeCallback(done, result);
|
||||
return;
|
||||
}
|
||||
_removing.erase(to);
|
||||
_stale.erase(ranges::remove(_stale, to), end(_stale));
|
||||
invokeCallback(done, writeExistingPlace(to, entry));
|
||||
}
|
||||
|
||||
void DatabaseObject::stats(FnMut<void(Stats&&)> &&done) {
|
||||
auto result = _taggedStats;
|
||||
auto zero = TaggedSummary();
|
||||
zero.count = _map.size();
|
||||
zero.totalSize = _totalSize;
|
||||
for (const auto &summary : result) {
|
||||
zero.count -= summary.second.count;
|
||||
zero.totalSize -= summary.second.totalSize;
|
||||
}
|
||||
result[0] = zero;
|
||||
invokeCallback(done, std::move(result));
|
||||
rpl::producer<Stats> DatabaseObject::stats() const {
|
||||
return _stats.events_starting_with(collectStats());
|
||||
}
|
||||
|
||||
Stats DatabaseObject::collectStats() const {
|
||||
auto result = Stats();
|
||||
result.tagged = _taggedStats;
|
||||
result.full.count = _map.size();
|
||||
result.full.totalSize = _totalSize;
|
||||
result.clearing = (_cleaner.object != nullptr) || !_stale.empty();
|
||||
return result;
|
||||
}
|
||||
|
||||
void DatabaseObject::writeBundlesLazy() {
|
||||
|
@ -1047,10 +1163,12 @@ void DatabaseObject::createCleaner() {
|
|||
_base,
|
||||
std::move(right),
|
||||
std::move(done));
|
||||
pushStatsDelayed();
|
||||
}
|
||||
|
||||
void DatabaseObject::cleanerDone(Error error) {
|
||||
_cleaner = CleanerWrap();
|
||||
pushStatsDelayed();
|
||||
}
|
||||
|
||||
void DatabaseObject::checkCompactor() {
|
||||
|
@ -1079,12 +1197,33 @@ void DatabaseObject::checkCompactor() {
|
|||
}
|
||||
|
||||
void DatabaseObject::clear(FnMut<void(Error)> &&done) {
|
||||
Expects(_key.empty());
|
||||
|
||||
auto key = std::move(_key);
|
||||
if (!key.empty()) {
|
||||
close(nullptr);
|
||||
}
|
||||
const auto version = findAvailableVersion();
|
||||
invokeCallback(
|
||||
done,
|
||||
writeVersion(version) ? Error::NoError() : ioError(versionPath()));
|
||||
if (!writeVersion(version)) {
|
||||
invokeCallback(done, ioError(versionPath()));
|
||||
return;
|
||||
}
|
||||
if (key.empty()) {
|
||||
invokeCallback(done, Error::NoError());
|
||||
return;
|
||||
}
|
||||
open(std::move(key), std::move(done));
|
||||
}
|
||||
|
||||
void DatabaseObject::clearByTag(uint8 tag, FnMut<void(Error)> &&done) {
|
||||
const auto hadStale = !_stale.empty();
|
||||
for (const auto &[key, entry] : _map) {
|
||||
if (entry.tag == tag) {
|
||||
_stale.push_back(key);
|
||||
}
|
||||
}
|
||||
if (!hadStale) {
|
||||
startStaleClear();
|
||||
}
|
||||
invokeCallback(done, Error::NoError());
|
||||
}
|
||||
|
||||
auto DatabaseObject::getManyRaw(const std::vector<Key> &keys) const
|
||||
|
|
|
@ -14,6 +14,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "base/bytes.h"
|
||||
#include "base/flat_set.h"
|
||||
#include <set>
|
||||
#include <rpl/event_stream.h>
|
||||
|
||||
namespace Storage {
|
||||
namespace Cache {
|
||||
|
@ -29,6 +30,7 @@ public:
|
|||
crl::weak_on_queue<DatabaseObject> weak,
|
||||
const QString &path,
|
||||
const Settings &settings);
|
||||
void reconfigure(const Settings &settings);
|
||||
|
||||
void open(EncryptionKey &&key, FnMut<void(Error)> &&done);
|
||||
void close(FnMut<void()> &&done);
|
||||
|
@ -53,9 +55,10 @@ public:
|
|||
const Key &to,
|
||||
FnMut<void(Error)> &&done);
|
||||
|
||||
void stats(FnMut<void(Stats&&)> &&done);
|
||||
rpl::producer<Stats> stats() const;
|
||||
|
||||
void clear(FnMut<void(Error)> &&done);
|
||||
void clearByTag(uint8 tag, FnMut<void(Error)> &&done);
|
||||
|
||||
static QString BinlogFilename();
|
||||
static QString CompactReadyFilename();
|
||||
|
@ -101,6 +104,7 @@ private:
|
|||
|
||||
Error ioError(const QString &path) const;
|
||||
|
||||
void checkSettings();
|
||||
QString computePath(Version version) const;
|
||||
QString binlogPath(Version version) const;
|
||||
QString binlogPath() const;
|
||||
|
@ -146,15 +150,24 @@ private:
|
|||
uint64 countRelativeTime() const;
|
||||
EstimatedTimePoint countTimePoint() const;
|
||||
void applyTimePoint(EstimatedTimePoint time);
|
||||
|
||||
uint64 pruneBeforeTime() const;
|
||||
void prune();
|
||||
void collectTimePrune(
|
||||
void collectTimeStale(
|
||||
base::flat_set<Key> &stale,
|
||||
int64 &staleTotalSize);
|
||||
void collectSizePrune(
|
||||
void collectSizeStale(
|
||||
base::flat_set<Key> &stale,
|
||||
int64 &staleTotalSize);
|
||||
void startStaleClear();
|
||||
void clearStaleNow(const base::flat_set<Key> &stale);
|
||||
void clearStaleChunkDelayed();
|
||||
void clearStaleChunk();
|
||||
|
||||
void updateStats(const Entry &was, const Entry &now);
|
||||
Stats collectStats() const;
|
||||
void pushStatsDelayed();
|
||||
void pushStats();
|
||||
|
||||
void setMapEntry(const Key &key, Entry &&entry);
|
||||
void eraseMapEntry(const Map::const_iterator &i);
|
||||
|
@ -197,15 +210,17 @@ private:
|
|||
|
||||
void createCleaner();
|
||||
void cleanerDone(Error error);
|
||||
void clearState();
|
||||
|
||||
crl::weak_on_queue<DatabaseObject> _weak;
|
||||
QString _base, _path;
|
||||
const Settings _settings;
|
||||
Settings _settings;
|
||||
EncryptionKey _key;
|
||||
File _binlog;
|
||||
Map _map;
|
||||
std::set<Key> _removing;
|
||||
std::set<Key> _accessed;
|
||||
std::vector<Key> _stale;
|
||||
|
||||
EstimatedTimePoint _time;
|
||||
|
||||
|
@ -214,7 +229,10 @@ private:
|
|||
uint64 _minimalEntryTime = 0;
|
||||
size_type _entriesWithMinimalTimeCount = 0;
|
||||
|
||||
Stats _taggedStats;
|
||||
base::flat_map<uint8, TaggedSummary> _taggedStats;
|
||||
rpl::event_stream<Stats> _stats;
|
||||
bool _pushingStats = false;
|
||||
bool _clearingStale = false;
|
||||
|
||||
base::ConcurrentTimer _writeBundlesTimer;
|
||||
base::ConcurrentTimer _pruneTimer;
|
||||
|
|
|
@ -56,6 +56,7 @@ struct Settings {
|
|||
size_type readBlockSize = 8 * 1024 * 1024;
|
||||
size_type maxDataSize = 10 * 1024 * 1024;
|
||||
crl::time_type writeBundleDelay = 15 * 60 * crl::time_type(1000);
|
||||
size_type staleRemoveChunk = 256;
|
||||
|
||||
int64 compactAfterExcess = 8 * 1024 * 1024;
|
||||
int64 compactAfterFullSize = 0;
|
||||
|
@ -66,6 +67,8 @@ struct Settings {
|
|||
size_type totalTimeLimit = 30 * 86400; // One month in seconds.
|
||||
crl::time_type pruneTimeout = 5 * crl::time_type(1000);
|
||||
crl::time_type maxPruneCheckTimeout = 3600 * crl::time_type(1000);
|
||||
|
||||
bool clearOnWrongKey = false;
|
||||
};
|
||||
|
||||
struct TaggedValue {
|
||||
|
@ -80,7 +83,11 @@ struct TaggedSummary {
|
|||
size_type count = 0;
|
||||
size_type totalSize = 0;
|
||||
};
|
||||
using Stats = base::flat_map<uint8, TaggedSummary>;
|
||||
struct Stats {
|
||||
TaggedSummary full;
|
||||
base::flat_map<uint8, TaggedSummary> tagged;
|
||||
bool clearing = false;
|
||||
};
|
||||
|
||||
using Version = int32;
|
||||
|
||||
|
|
|
@ -2949,6 +2949,7 @@ QString cachePath() {
|
|||
|
||||
Storage::Cache::Database::Settings cacheSettings() {
|
||||
auto result = Storage::Cache::Database::Settings();
|
||||
result.clearOnWrongKey = true;
|
||||
return result;
|
||||
}
|
||||
|
||||
|
@ -2958,7 +2959,6 @@ Storage::EncryptionKey cacheKey() {
|
|||
return Storage::EncryptionKey(bytes::make_vector(LocalKey->data()));
|
||||
}
|
||||
|
||||
|
||||
class CountWaveformTask : public Task {
|
||||
public:
|
||||
CountWaveformTask(DocumentData *doc)
|
||||
|
|
|
@ -0,0 +1,103 @@
|
|||
/*
|
||||
This file is part of Telegram Desktop,
|
||||
the official desktop application for the Telegram messaging service.
|
||||
|
||||
For license and copyright information please follow this link:
|
||||
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||
*/
|
||||
#include "storage/storage_databases.h"
|
||||
|
||||
#include "storage/cache/storage_cache_database.h"
|
||||
|
||||
namespace Storage {
|
||||
|
||||
DatabasePointer::DatabasePointer(
|
||||
not_null<Databases*> owner,
|
||||
const std::unique_ptr<Cache::Database> &value)
|
||||
: _owner(owner)
|
||||
, _value(value.get()) {
|
||||
}
|
||||
|
||||
DatabasePointer::DatabasePointer(DatabasePointer &&other)
|
||||
: _value(base::take(other._value))
|
||||
, _owner(other._owner) {
|
||||
}
|
||||
|
||||
DatabasePointer &DatabasePointer::operator=(DatabasePointer &&other) {
|
||||
if (this != &other) {
|
||||
destroy();
|
||||
_owner = other._owner;
|
||||
_value = base::take(other._value);
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
DatabasePointer::~DatabasePointer() {
|
||||
destroy();
|
||||
}
|
||||
|
||||
Cache::Database *DatabasePointer::get() const {
|
||||
return _value;
|
||||
}
|
||||
|
||||
Cache::Database &DatabasePointer::operator*() const {
|
||||
Expects(_value != nullptr);
|
||||
|
||||
return *get();
|
||||
}
|
||||
|
||||
Cache::Database *DatabasePointer::operator->() const {
|
||||
Expects(_value != nullptr);
|
||||
|
||||
return get();
|
||||
}
|
||||
|
||||
DatabasePointer::operator bool() const {
|
||||
return get() != nullptr;
|
||||
}
|
||||
|
||||
void DatabasePointer::destroy() {
|
||||
if (const auto value = base::take(_value)) {
|
||||
_owner->destroy(value);
|
||||
}
|
||||
}
|
||||
|
||||
Databases::Kept::Kept(std::unique_ptr<Cache::Database> &&database)
|
||||
: database(std::move(database)) {
|
||||
}
|
||||
|
||||
DatabasePointer Databases::get(
|
||||
const QString &path,
|
||||
const Cache::details::Settings &settings) {
|
||||
if (const auto i = _map.find(path); i != end(_map)) {
|
||||
auto &kept = i->second;
|
||||
Assert(kept.destroying.alive());
|
||||
kept.destroying.kill();
|
||||
kept.database->reconfigure(settings);
|
||||
return DatabasePointer(this, kept.database);
|
||||
}
|
||||
const auto [i, ok] = _map.emplace(
|
||||
path,
|
||||
std::make_unique<Cache::Database>(path, settings));
|
||||
return DatabasePointer(this, i->second.database);
|
||||
}
|
||||
|
||||
void Databases::destroy(Cache::Database *database) {
|
||||
for (auto &[path, kept] : _map) {
|
||||
if (kept.database.get() == database) {
|
||||
Assert(!kept.destroying.alive());
|
||||
auto [first, second] = base::make_binary_guard();
|
||||
kept.destroying = std::move(first);
|
||||
database->close([=, guard = std::move(second)]() mutable {
|
||||
crl::on_main([=, guard = std::move(guard)]{
|
||||
if (!guard.alive()) {
|
||||
return;
|
||||
}
|
||||
_map.erase(path);
|
||||
});
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
} // namespace Storage
|
|
@ -0,0 +1,71 @@
|
|||
/*
|
||||
This file is part of Telegram Desktop,
|
||||
the official desktop application for the Telegram messaging service.
|
||||
|
||||
For license and copyright information please follow this link:
|
||||
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
#include "storage/cache/storage_cache_database.h"
|
||||
#include "base/binary_guard.h"
|
||||
|
||||
namespace Storage {
|
||||
namespace Cache {
|
||||
namespace details {
|
||||
struct Settings;
|
||||
} // namespace details
|
||||
class Database;
|
||||
} // namespace Cache
|
||||
|
||||
class Databases;
|
||||
|
||||
class DatabasePointer {
|
||||
public:
|
||||
DatabasePointer(const DatabasePointer &other) = delete;
|
||||
DatabasePointer(DatabasePointer &&other);
|
||||
DatabasePointer &operator=(const DatabasePointer &other) = delete;
|
||||
DatabasePointer &operator=(DatabasePointer &&other);
|
||||
~DatabasePointer();
|
||||
|
||||
Cache::Database *get() const;
|
||||
Cache::Database &operator*() const;
|
||||
Cache::Database *operator->() const;
|
||||
explicit operator bool() const;
|
||||
|
||||
private:
|
||||
friend class Databases;
|
||||
|
||||
DatabasePointer(
|
||||
not_null<Databases*> owner,
|
||||
const std::unique_ptr<Cache::Database> &value);
|
||||
void destroy();
|
||||
|
||||
Cache::Database *_value = nullptr;
|
||||
not_null<Databases*> _owner;
|
||||
|
||||
};
|
||||
|
||||
class Databases {
|
||||
public:
|
||||
DatabasePointer get(
|
||||
const QString &path,
|
||||
const Cache::details::Settings &settings);
|
||||
|
||||
private:
|
||||
friend class DatabasePointer;
|
||||
|
||||
struct Kept {
|
||||
Kept(std::unique_ptr<Cache::Database> &&database);
|
||||
|
||||
std::unique_ptr<Cache::Database> database;
|
||||
base::binary_guard destroying;
|
||||
};
|
||||
|
||||
void destroy(Cache::Database *database);
|
||||
|
||||
std::map<QString, Kept> _map;
|
||||
|
||||
};
|
||||
|
||||
} // namespace Storage
|
|
@ -100,7 +100,10 @@ private:
|
|||
|
||||
class RoundButton : public RippleButton, private base::Subscriber {
|
||||
public:
|
||||
RoundButton(QWidget *parent, Fn<QString()> textFactory, const style::RoundButton &st);
|
||||
RoundButton(
|
||||
QWidget *parent,
|
||||
Fn<QString()> textFactory,
|
||||
const style::RoundButton &st);
|
||||
|
||||
void setText(Fn<QString()> textFactory);
|
||||
|
||||
|
|
|
@ -50,6 +50,8 @@
|
|||
'<(submodules_loc)/xxHash',
|
||||
],
|
||||
'sources': [
|
||||
'<(src_loc)/storage/storage_databases.cpp',
|
||||
'<(src_loc)/storage/storage_databases.h',
|
||||
'<(src_loc)/storage/storage_encryption.cpp',
|
||||
'<(src_loc)/storage/storage_encryption.h',
|
||||
'<(src_loc)/storage/storage_encrypted_file.cpp',
|
||||
|
|
Loading…
Reference in New Issue