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_settings_section_privacy" = "Privacy and Security"; | ||||||
| 
 | 
 | ||||||
| "lng_local_storage_title" = "Local storage"; | "lng_local_storage_title" = "Local storage"; | ||||||
| "lng_settings_no_data_cached" = "No cached data found!"; | "lng_local_storage_empty" = "No cached files"; | ||||||
| "lng_settings_images_cached#one" = "{count} image, {size}"; | "lng_local_storage_image#one" = "{count} image"; | ||||||
| "lng_settings_images_cached#other" = "{count} images, {size}"; | "lng_local_storage_image#other" = "{count} images"; | ||||||
| "lng_settings_audios_cached#one" = "{count} voice message, {size}"; | "lng_local_storage_sticker#one" = "{count} sticker"; | ||||||
| "lng_settings_audios_cached#other" = "{count} voice messages, {size}"; | "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_clear" = "Clear all"; | ||||||
| "lng_local_storage_clearing" = "Clearing..."; | "lng_local_storage_clearing" = "Clearing..."; | ||||||
| "lng_local_storage_cleared" = "Cleared!"; |  | ||||||
| 
 | 
 | ||||||
| "lng_settings_section_advanced_settings" = "Advanced Settings"; | "lng_settings_section_advanced_settings" = "Advanced Settings"; | ||||||
| "lng_settings_enable_night_theme" = "Enable night mode"; | "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 { | 	size_type size() const { | ||||||
| 		return impl().size(); | 		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]") { | TEST_CASE("flat_maps custom comparator", "[flat_map]") { | ||||||
| 	base::flat_map<int_wrap, string, int_wrap_comparator> v; | 	base::flat_map<int_wrap, string, int_wrap_comparator> v; | ||||||
| 	v.emplace({ 0 }, "a"); | 	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; | shareRowsTop: 12px; | ||||||
| shareRowHeight: 108px; | shareRowHeight: 108px; | ||||||
|  |  | ||||||
|  | @ -8,15 +8,185 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| #include "boxes/local_storage_box.h" | #include "boxes/local_storage_box.h" | ||||||
| 
 | 
 | ||||||
| #include "styles/style_boxes.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/widgets/buttons.h" | ||||||
|  | #include "ui/effects/radial_animation.h" | ||||||
|  | #include "ui/widgets/shadow.h" | ||||||
| #include "storage/localstorage.h" | #include "storage/localstorage.h" | ||||||
| #include "lang/lang_keys.h" | #include "lang/lang_keys.h" | ||||||
| #include "mainwindow.h" | #include "mainwindow.h" | ||||||
| #include "auth_session.h" | #include "auth_session.h" | ||||||
| #include "layout.h" | #include "layout.h" | ||||||
| 
 | 
 | ||||||
| LocalStorageBox::LocalStorageBox(QWidget *parent) | class LocalStorageBox::Row : public Ui::RpWidget { | ||||||
| : _clear(this, lang(lng_local_storage_clear), st::boxLinkButton) { | 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() { | void LocalStorageBox::prepare() { | ||||||
|  | @ -24,27 +194,110 @@ void LocalStorageBox::prepare() { | ||||||
| 
 | 
 | ||||||
| 	addButton(langFactory(lng_box_ok), [this] { closeBox(); }); | 	addButton(langFactory(lng_box_ok), [this] { closeBox(); }); | ||||||
| 
 | 
 | ||||||
| 	_clear->setClickedCallback([this] { clearStorage(); }); | 	setupControls(); | ||||||
| 
 |  | ||||||
| 	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(); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void LocalStorageBox::updateControls() { | void LocalStorageBox::updateRow( | ||||||
| 	const auto rowsHeight = st::linkFont->height + st::localStorageBoxSkip; | 		not_null<Ui::SlideWrap<Row>*> row, | ||||||
| 	_clear->setVisible(false); | 		Database::TaggedSummary *data) { | ||||||
| 	setDimensions(st::boxWidth, st::localStorageBoxSkip + rowsHeight + _clear->height()); | 	const auto summary = (_rows.find(0)->second == row); | ||||||
| 	_clear->moveToLeft(st::boxPadding.left(), st::localStorageBoxSkip + rowsHeight); | 	const auto shown = (data && data->count && data->totalSize) || summary; | ||||||
| 	update(); | 	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) { | void LocalStorageBox::paintEvent(QPaintEvent *e) { | ||||||
|  | @ -54,40 +307,4 @@ void LocalStorageBox::paintEvent(QPaintEvent *e) { | ||||||
| 
 | 
 | ||||||
| 	p.setFont(st::boxTextFont); | 	p.setFont(st::boxTextFont); | ||||||
| 	p.setPen(st::windowFg); | 	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 | #pragma once | ||||||
| 
 | 
 | ||||||
| #include "boxes/abstract_box.h" | #include "boxes/abstract_box.h" | ||||||
|  | #include "storage/cache/storage_cache_database.h" | ||||||
|  | 
 | ||||||
|  | namespace Storage { | ||||||
|  | namespace Cache { | ||||||
|  | class Database; | ||||||
|  | } // namespace Cache
 | ||||||
|  | } // namespace Storage
 | ||||||
| 
 | 
 | ||||||
| namespace Ui { | namespace Ui { | ||||||
| class LinkButton; | class VerticalLayout; | ||||||
|  | template <typename Widget> | ||||||
|  | class SlideWrap; | ||||||
| } // namespace Ui
 | } // namespace Ui
 | ||||||
| 
 | 
 | ||||||
| class LocalStorageBox : public BoxContent { | class LocalStorageBox : public BoxContent { | ||||||
| 	Q_OBJECT | 	struct CreateTag { | ||||||
|  | 	}; | ||||||
| 
 | 
 | ||||||
| public: | public: | ||||||
| 	LocalStorageBox(QWidget*); | 	using Database = Storage::Cache::Database; | ||||||
| 
 | 
 | ||||||
| private slots: | 	LocalStorageBox(QWidget*, not_null<Database*> db, CreateTag); | ||||||
| 	void onTempDirCleared(int task); | 
 | ||||||
| 	void onTempDirClearFailed(int task); | 	static void Show(not_null<Database*> db); | ||||||
| 
 | 
 | ||||||
| protected: | protected: | ||||||
| 	void prepare() override; | 	void prepare() override; | ||||||
|  | @ -29,18 +39,19 @@ protected: | ||||||
| 	void paintEvent(QPaintEvent *e) override; | 	void paintEvent(QPaintEvent *e) override; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
| 	void clearStorage(); | 	class Row; | ||||||
| 	void updateControls(); |  | ||||||
| 	void checkLocalStoredCounts(); |  | ||||||
| 
 | 
 | ||||||
| 	enum class State { | 	void clearByTag(uint8 tag); | ||||||
| 		Normal, | 	void update(Database::Stats &&stats); | ||||||
| 		Clearing, | 	void updateRow( | ||||||
| 		Cleared, | 		not_null<Ui::SlideWrap<Row>*> row, | ||||||
| 		ClearFailed, | 		Database::TaggedSummary *data); | ||||||
| 	}; | 	void setupControls(); | ||||||
| 	State _state = State::Normal; |  | ||||||
| 
 | 
 | ||||||
| 	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 "observer_peer.h" | ||||||
| #include "auth_session.h" | #include "auth_session.h" | ||||||
| #include "apiwrap.h" | #include "apiwrap.h" | ||||||
|  | #include "messenger.h" | ||||||
| #include "export/export_controller.h" | #include "export/export_controller.h" | ||||||
| #include "export/view/export_view_panel_controller.h" | #include "export/view/export_view_panel_controller.h" | ||||||
| #include "window/notifications_manager.h" | #include "window/notifications_manager.h" | ||||||
|  | @ -67,17 +68,19 @@ void UpdateImage(ImagePtr &old, ImagePtr now) { | ||||||
| 
 | 
 | ||||||
| Session::Session(not_null<AuthSession*> session) | Session::Session(not_null<AuthSession*> session) | ||||||
| : _session(session) | : _session(session) | ||||||
| , _cache(Local::cachePath(), Local::cacheSettings()) | , _cache(Messenger::Instance().databases().get( | ||||||
|  | 	Local::cachePath(), | ||||||
|  | 	Local::cacheSettings())) | ||||||
| , _groups(this) | , _groups(this) | ||||||
| , _unmuteByFinishedTimer([=] { unmuteByFinished(); }) { | , _unmuteByFinishedTimer([=] { unmuteByFinished(); }) { | ||||||
| 	_cache.open(Local::cacheKey()); | 	_cache->open(Local::cacheKey()); | ||||||
| 
 | 
 | ||||||
| 	setupContactViewsViewer(); | 	setupContactViewsViewer(); | ||||||
| 	setupChannelLeavingViewer(); | 	setupChannelLeavingViewer(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Storage::Cache::Database &Session::cache() { | Storage::Cache::Database &Session::cache() { | ||||||
| 	return _cache; | 	return *_cache; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Session::startExport(PeerData *peer) { | void Session::startExport(PeerData *peer) { | ||||||
|  |  | ||||||
|  | @ -7,7 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| */ | */ | ||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include "storage/cache/storage_cache_database.h" | #include "storage/storage_databases.h" | ||||||
| #include "chat_helpers/stickers.h" | #include "chat_helpers/stickers.h" | ||||||
| #include "dialogs/dialogs_key.h" | #include "dialogs/dialogs_key.h" | ||||||
| #include "data/data_groups.h" | #include "data/data_groups.h" | ||||||
|  | @ -523,7 +523,7 @@ private: | ||||||
| 
 | 
 | ||||||
| 	not_null<AuthSession*> _session; | 	not_null<AuthSession*> _session; | ||||||
| 
 | 
 | ||||||
| 	Storage::Cache::Database _cache; | 	Storage::DatabasePointer _cache; | ||||||
| 
 | 
 | ||||||
| 	std::unique_ptr<Export::ControllerWrap> _export; | 	std::unique_ptr<Export::ControllerWrap> _export; | ||||||
| 	std::unique_ptr<Export::View::PanelController> _exportPanel; | 	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 "passport/passport_form_controller.h" | ||||||
| #include "observer_peer.h" | #include "observer_peer.h" | ||||||
| #include "storage/file_upload.h" | #include "storage/file_upload.h" | ||||||
|  | #include "storage/storage_databases.h" | ||||||
| #include "mainwidget.h" | #include "mainwidget.h" | ||||||
| #include "mediaview.h" | #include "mediaview.h" | ||||||
| #include "mtproto/dc_options.h" | #include "mtproto/dc_options.h" | ||||||
|  | @ -76,6 +77,7 @@ Messenger::Messenger(not_null<Core::Launcher*> launcher) | ||||||
| : QObject() | : QObject() | ||||||
| , _launcher(launcher) | , _launcher(launcher) | ||||||
| , _private(std::make_unique<Private>()) | , _private(std::make_unique<Private>()) | ||||||
|  | , _databases(std::make_unique<Storage::Databases>()) | ||||||
| , _langpack(std::make_unique<Lang::Instance>()) | , _langpack(std::make_unique<Lang::Instance>()) | ||||||
| , _audio(std::make_unique<Media::Audio::Instance>()) | , _audio(std::make_unique<Media::Audio::Instance>()) | ||||||
| , _logo(Window::LoadLogo()) | , _logo(Window::LoadLogo()) | ||||||
|  |  | ||||||
|  | @ -19,6 +19,10 @@ class Translator; | ||||||
| class MediaView; | class MediaView; | ||||||
| class BoxContent; | class BoxContent; | ||||||
| 
 | 
 | ||||||
|  | namespace Storage { | ||||||
|  | class Databases; | ||||||
|  | } // namespace Storage
 | ||||||
|  | 
 | ||||||
| namespace Core { | namespace Core { | ||||||
| class Launcher; | class Launcher; | ||||||
| } // namespace Core
 | } // namespace Core
 | ||||||
|  | @ -125,6 +129,11 @@ public: | ||||||
| 	void suggestMainDcId(MTP::DcId mainDcId); | 	void suggestMainDcId(MTP::DcId mainDcId); | ||||||
| 	void destroyStaleAuthorizationKeys(); | 	void destroyStaleAuthorizationKeys(); | ||||||
| 
 | 
 | ||||||
|  | 	// Databases
 | ||||||
|  | 	Storage::Databases &databases() { | ||||||
|  | 		return *_databases; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	// AuthSession component.
 | 	// AuthSession component.
 | ||||||
| 	AuthSession *authSession() { | 	AuthSession *authSession() { | ||||||
| 		return _authSession.get(); | 		return _authSession.get(); | ||||||
|  | @ -249,6 +258,7 @@ private: | ||||||
| 
 | 
 | ||||||
| 	QWidget _globalShortcutParent; | 	QWidget _globalShortcutParent; | ||||||
| 
 | 
 | ||||||
|  | 	std::unique_ptr<Storage::Databases> _databases; | ||||||
| 	std::unique_ptr<MainWindow> _window; | 	std::unique_ptr<MainWindow> _window; | ||||||
| 	std::unique_ptr<MediaView> _mediaView; | 	std::unique_ptr<MediaView> _mediaView; | ||||||
| 	std::unique_ptr<Lang::Instance> _langpack; | 	std::unique_ptr<Lang::Instance> _langpack; | ||||||
|  |  | ||||||
|  | @ -34,7 +34,7 @@ public: | ||||||
| 	void fire_copy(const Value &value) const { | 	void fire_copy(const Value &value) const { | ||||||
| 		return fire_forward(value); | 		return fire_forward(value); | ||||||
| 	} | 	} | ||||||
| #if defined _MSC_VER && _MSC_VER >= 1914 | #if defined _MSC_VER && _MSC_VER >= 1914 && false | ||||||
| 	producer<Value> events() const { | 	producer<Value> events() const { | ||||||
| #else // _MSC_VER >= 1914
 | #else // _MSC_VER >= 1914
 | ||||||
| 	auto events() const { | 	auto events() const { | ||||||
|  | @ -65,6 +65,9 @@ public: | ||||||
| 	auto events_starting_with_copy(const Value &value) const { | 	auto events_starting_with_copy(const Value &value) const { | ||||||
| 		return single(value) | then(events()); | 		return single(value) | then(events()); | ||||||
| 	} | 	} | ||||||
|  | 	bool has_consumers() const { | ||||||
|  | 		return (_data != nullptr) && !_data->consumers.empty(); | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	~event_stream(); | 	~event_stream(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| #include "boxes/confirm_box.h" | #include "boxes/confirm_box.h" | ||||||
| #include "boxes/about_box.h" | #include "boxes/about_box.h" | ||||||
| #include "boxes/local_storage_box.h" | #include "boxes/local_storage_box.h" | ||||||
|  | #include "data/data_session.h" | ||||||
| #include "mainwindow.h" | #include "mainwindow.h" | ||||||
| #include "ui/widgets/buttons.h" | #include "ui/widgets/buttons.h" | ||||||
| #include "ui/wrap/slide_wrap.h" | #include "ui/wrap/slide_wrap.h" | ||||||
|  | @ -85,7 +86,7 @@ void AdvancedWidget::checkNonDefaultTheme() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void AdvancedWidget::onManageLocalStorage() { | void AdvancedWidget::onManageLocalStorage() { | ||||||
| 	Ui::show(Box<LocalStorageBox>()); | 	LocalStorageBox::Show(&Auth().data().cache()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #ifndef TDESKTOP_DISABLE_NETWORK_PROXY | #ifndef TDESKTOP_DISABLE_NETWORK_PROXY | ||||||
|  |  | ||||||
|  | @ -16,6 +16,12 @@ Database::Database(const QString &path, const Settings &settings) | ||||||
| : _wrapped(path, 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) { | void Database::open(EncryptionKey &&key, FnMut<void(Error)> &&done) { | ||||||
| 	_wrapped.with([ | 	_wrapped.with([ | ||||||
| 		key = std::move(key), | 		key = std::move(key), | ||||||
|  | @ -125,7 +131,7 @@ void Database::putIfEmpty( | ||||||
| void Database::getWithTag( | void Database::getWithTag( | ||||||
| 		const Key &key, | 		const Key &key, | ||||||
| 		FnMut<void(TaggedValue&&)> &&done) { | 		FnMut<void(TaggedValue&&)> &&done) { | ||||||
| 		_wrapped.with([ | 	_wrapped.with([ | ||||||
| 		key, | 		key, | ||||||
| 		done = std::move(done) | 		done = std::move(done) | ||||||
| 	](Implementation &unwrapped) mutable { | 	](Implementation &unwrapped) mutable { | ||||||
|  | @ -133,11 +139,9 @@ void Database::getWithTag( | ||||||
| 	}); | 	}); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Database::stats(FnMut<void(Stats&&)> &&done) { | auto Database::statsOnMain() const -> rpl::producer<Stats> { | ||||||
| 	_wrapped.with([ | 	return _wrapped.producer_on_main([](const Implementation &unwrapped) { | ||||||
| 		done = std::move(done) | 		return unwrapped.stats(); | ||||||
| 	](Implementation &unwrapped) mutable { |  | ||||||
| 		unwrapped.stats(std::move(done)); |  | ||||||
| 	}); | 	}); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -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; | Database::~Database() = default; | ||||||
| 
 | 
 | ||||||
| } // namespace Cache
 | } // namespace Cache
 | ||||||
|  |  | ||||||
|  | @ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| #include "base/basic_types.h" | #include "base/basic_types.h" | ||||||
| #include <crl/crl_object_on_queue.h> | #include <crl/crl_object_on_queue.h> | ||||||
| #include <crl/crl_time.h> | #include <crl/crl_time.h> | ||||||
|  | #include <rpl/producer.h> | ||||||
| #include <QtCore/QString> | #include <QtCore/QString> | ||||||
| 
 | 
 | ||||||
| namespace Storage { | namespace Storage { | ||||||
|  | @ -25,6 +26,8 @@ public: | ||||||
| 	using Settings = details::Settings; | 	using Settings = details::Settings; | ||||||
| 	Database(const QString &path, const Settings &settings); | 	Database(const QString &path, const Settings &settings); | ||||||
| 
 | 
 | ||||||
|  | 	void reconfigure(const Settings &settings); | ||||||
|  | 
 | ||||||
| 	void open(EncryptionKey &&key, FnMut<void(Error)> &&done = nullptr); | 	void open(EncryptionKey &&key, FnMut<void(Error)> &&done = nullptr); | ||||||
| 	void close(FnMut<void()> &&done = nullptr); | 	void close(FnMut<void()> &&done = nullptr); | ||||||
| 
 | 
 | ||||||
|  | @ -60,9 +63,11 @@ public: | ||||||
| 	void getWithTag(const Key &key, FnMut<void(TaggedValue&&)> &&done); | 	void getWithTag(const Key &key, FnMut<void(TaggedValue&&)> &&done); | ||||||
| 
 | 
 | ||||||
| 	using Stats = details::Stats; | 	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 clear(FnMut<void(Error)> &&done = nullptr); | ||||||
|  | 	void clearByTag(uint8 tag, FnMut<void(Error)> &&done = nullptr); | ||||||
| 
 | 
 | ||||||
| 	~Database(); | 	~Database(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -82,6 +82,18 @@ DatabaseObject::DatabaseObject( | ||||||
| , _settings(settings) | , _settings(settings) | ||||||
| , _writeBundlesTimer(_weak, [=] { writeBundles(); checkCompactor(); }) | , _writeBundlesTimer(_weak, [=] { writeBundles(); checkCompactor(); }) | ||||||
| , _pruneTimer(_weak, [=] { prune(); }) { | , _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 | 	Expects(_settings.maxDataSize > 0 | ||||||
| 		&& _settings.maxDataSize < kDataSizeLimit); | 		&& _settings.maxDataSize < kDataSizeLimit); | ||||||
| 	Expects(_settings.maxBundledRecords > 0 | 	Expects(_settings.maxBundledRecords > 0 | ||||||
|  | @ -124,7 +136,9 @@ Error DatabaseObject::openSomeBinlog(EncryptionKey &&key) { | ||||||
| 	case File::Result::LockFailed: | 	case File::Result::LockFailed: | ||||||
| 		return Error{ Error::Type::LockFailed, binlogPath(version) }; | 		return Error{ Error::Type::LockFailed, binlogPath(version) }; | ||||||
| 	case File::Result::WrongKey: | 	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."); | 	Unexpected("Result from DatabaseObject::openBinlog."); | ||||||
| } | } | ||||||
|  | @ -290,8 +304,8 @@ bool DatabaseObject::startDelayedPruning() { | ||||||
| 		if (_settings.totalSizeLimit > 0 | 		if (_settings.totalSizeLimit > 0 | ||||||
| 			&& _totalSize > _settings.totalSizeLimit) { | 			&& _totalSize > _settings.totalSizeLimit) { | ||||||
| 			return true; | 			return true; | ||||||
| 		} else if (_minimalEntryTime != 0 | 		} else if ((!_minimalEntryTime && !_map.empty()) | ||||||
| 			&& _minimalEntryTime <= before) { | 			|| _minimalEntryTime <= before) { | ||||||
| 			return true; | 			return true; | ||||||
| 		} | 		} | ||||||
| 		return false; | 		return false; | ||||||
|  | @ -315,17 +329,77 @@ bool DatabaseObject::startDelayedPruning() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DatabaseObject::prune() { | void DatabaseObject::prune() { | ||||||
|  | 	if (!_stale.empty()) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
| 	auto stale = base::flat_set<Key>(); | 	auto stale = base::flat_set<Key>(); | ||||||
| 	auto staleTotalSize = int64(); | 	auto staleTotalSize = int64(); | ||||||
| 	collectTimePrune(stale, staleTotalSize); | 	collectTimeStale(stale, staleTotalSize); | ||||||
| 	collectSizePrune(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) { | 	for (const auto &key : stale) { | ||||||
| 		remove(key, nullptr); | 		remove(key, nullptr); | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	// Report correct status async.
 | ||||||
|  | 	_stale.clear(); | ||||||
| 	optimize(); | 	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, | 		base::flat_set<Key> &stale, | ||||||
| 		int64 &staleTotalSize) { | 		int64 &staleTotalSize) { | ||||||
| 	if (!_settings.totalTimeLimit) { | 	if (!_settings.totalTimeLimit) { | ||||||
|  | @ -351,7 +425,7 @@ void DatabaseObject::collectTimePrune( | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DatabaseObject::collectSizePrune( | void DatabaseObject::collectSizeStale( | ||||||
| 		base::flat_set<Key> &stale, | 		base::flat_set<Key> &stale, | ||||||
| 		int64 &staleTotalSize) { | 		int64 &staleTotalSize) { | ||||||
| 	const auto removeSize = (_settings.totalSizeLimit > 0) | 	const auto removeSize = (_settings.totalSizeLimit > 0) | ||||||
|  | @ -516,7 +590,9 @@ void DatabaseObject::setMapEntry(const Key &key, Entry &&entry) { | ||||||
| 			++_entriesWithMinimalTimeCount; | 			++_entriesWithMinimalTimeCount; | ||||||
| 		} else if (already.useTime == _minimalEntryTime) { | 		} else if (already.useTime == _minimalEntryTime) { | ||||||
| 			Assert(_entriesWithMinimalTimeCount > 0); | 			Assert(_entriesWithMinimalTimeCount > 0); | ||||||
| 			--_entriesWithMinimalTimeCount; | 			if (!--_entriesWithMinimalTimeCount) { | ||||||
|  | 				_minimalEntryTime = 0; | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	already = std::move(entry); | 	already = std::move(entry); | ||||||
|  | @ -542,6 +618,25 @@ void DatabaseObject::updateStats(const Entry &was, const Entry &now) { | ||||||
| 			summary.totalSize -= was.size; | 			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) { | void DatabaseObject::eraseMapEntry(const Map::const_iterator &i) { | ||||||
|  | @ -550,7 +645,9 @@ void DatabaseObject::eraseMapEntry(const Map::const_iterator &i) { | ||||||
| 		updateStats(entry, Entry()); | 		updateStats(entry, Entry()); | ||||||
| 		if (_minimalEntryTime != 0 && entry.useTime == _minimalEntryTime) { | 		if (_minimalEntryTime != 0 && entry.useTime == _minimalEntryTime) { | ||||||
| 			Assert(_entriesWithMinimalTimeCount > 0); | 			Assert(_entriesWithMinimalTimeCount > 0); | ||||||
| 			--_entriesWithMinimalTimeCount; | 			if (!--_entriesWithMinimalTimeCount) { | ||||||
|  | 				_minimalEntryTime = 0; | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		_map.erase(i); | 		_map.erase(i); | ||||||
| 	} | 	} | ||||||
|  | @ -629,14 +726,29 @@ void DatabaseObject::compactorFail() { | ||||||
| void DatabaseObject::close(FnMut<void()> &&done) { | void DatabaseObject::close(FnMut<void()> &&done) { | ||||||
| 	if (_binlog.isOpen()) { | 	if (_binlog.isOpen()) { | ||||||
| 		writeBundles(); | 		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(); | 	_cleaner = CleanerWrap(); | ||||||
| 	_compactor = CompactorWrap(); | 	_compactor = CompactorWrap(); | ||||||
| 	_binlog.close(); |  | ||||||
| 	_key = EncryptionKey(); |  | ||||||
| 	invokeCallback(done); |  | ||||||
| 	_map.clear(); |  | ||||||
| 	_binlogExcessLength = 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DatabaseObject::put( | void DatabaseObject::put( | ||||||
|  | @ -648,6 +760,7 @@ void DatabaseObject::put( | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 	_removing.erase(key); | 	_removing.erase(key); | ||||||
|  | 	_stale.erase(ranges::remove(_stale, key), end(_stale)); | ||||||
| 
 | 
 | ||||||
| 	const auto checksum = CountChecksum(bytes::make_span(value.bytes)); | 	const auto checksum = CountChecksum(bytes::make_span(value.bytes)); | ||||||
| 	const auto maybepath = writeKeyPlace(key, value, checksum); | 	const auto maybepath = writeKeyPlace(key, value, checksum); | ||||||
|  | @ -762,6 +875,7 @@ Error DatabaseObject::writeExistingPlaceGeneric( | ||||||
| 		const Key &key, | 		const Key &key, | ||||||
| 		const Entry &entry) { | 		const Entry &entry) { | ||||||
| 	record.key = key; | 	record.key = key; | ||||||
|  | 	record.tag = entry.tag; | ||||||
| 	record.size = ReadTo<EntrySize>(entry.size); | 	record.size = ReadTo<EntrySize>(entry.size); | ||||||
| 	record.checksum = entry.checksum; | 	record.checksum = entry.checksum; | ||||||
| 	if (const auto i = _map.find(key); i != end(_map)) { | 	if (const auto i = _map.find(key); i != end(_map)) { | ||||||
|  | @ -927,20 +1041,22 @@ void DatabaseObject::moveIfEmpty( | ||||||
| 		invokeCallback(done, result); | 		invokeCallback(done, result); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
|  | 	_removing.erase(to); | ||||||
|  | 	_stale.erase(ranges::remove(_stale, to), end(_stale)); | ||||||
| 	invokeCallback(done, writeExistingPlace(to, entry)); | 	invokeCallback(done, writeExistingPlace(to, entry)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DatabaseObject::stats(FnMut<void(Stats&&)> &&done) { | rpl::producer<Stats> DatabaseObject::stats() const { | ||||||
| 	auto result = _taggedStats; | 	return _stats.events_starting_with(collectStats()); | ||||||
| 	auto zero = TaggedSummary(); | } | ||||||
| 	zero.count = _map.size(); | 
 | ||||||
| 	zero.totalSize = _totalSize; | Stats DatabaseObject::collectStats() const { | ||||||
| 	for (const auto &summary : result) { | 	auto result = Stats(); | ||||||
| 		zero.count -= summary.second.count; | 	result.tagged = _taggedStats; | ||||||
| 		zero.totalSize -= summary.second.totalSize; | 	result.full.count = _map.size(); | ||||||
| 	} | 	result.full.totalSize = _totalSize; | ||||||
| 	result[0] = zero; | 	result.clearing = (_cleaner.object != nullptr) || !_stale.empty(); | ||||||
| 	invokeCallback(done, std::move(result)); | 	return result; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DatabaseObject::writeBundlesLazy() { | void DatabaseObject::writeBundlesLazy() { | ||||||
|  | @ -1047,10 +1163,12 @@ void DatabaseObject::createCleaner() { | ||||||
| 		_base, | 		_base, | ||||||
| 		std::move(right), | 		std::move(right), | ||||||
| 		std::move(done)); | 		std::move(done)); | ||||||
|  | 	pushStatsDelayed(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DatabaseObject::cleanerDone(Error error) { | void DatabaseObject::cleanerDone(Error error) { | ||||||
| 	_cleaner = CleanerWrap(); | 	_cleaner = CleanerWrap(); | ||||||
|  | 	pushStatsDelayed(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DatabaseObject::checkCompactor() { | void DatabaseObject::checkCompactor() { | ||||||
|  | @ -1079,12 +1197,33 @@ void DatabaseObject::checkCompactor() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DatabaseObject::clear(FnMut<void(Error)> &&done) { | void DatabaseObject::clear(FnMut<void(Error)> &&done) { | ||||||
| 	Expects(_key.empty()); | 	auto key = std::move(_key); | ||||||
| 
 | 	if (!key.empty()) { | ||||||
|  | 		close(nullptr); | ||||||
|  | 	} | ||||||
| 	const auto version = findAvailableVersion(); | 	const auto version = findAvailableVersion(); | ||||||
| 	invokeCallback( | 	if (!writeVersion(version)) { | ||||||
| 		done, | 		invokeCallback(done, ioError(versionPath())); | ||||||
| 		writeVersion(version) ? Error::NoError() : 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 | 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/bytes.h" | ||||||
| #include "base/flat_set.h" | #include "base/flat_set.h" | ||||||
| #include <set> | #include <set> | ||||||
|  | #include <rpl/event_stream.h> | ||||||
| 
 | 
 | ||||||
| namespace Storage { | namespace Storage { | ||||||
| namespace Cache { | namespace Cache { | ||||||
|  | @ -29,6 +30,7 @@ public: | ||||||
| 		crl::weak_on_queue<DatabaseObject> weak, | 		crl::weak_on_queue<DatabaseObject> weak, | ||||||
| 		const QString &path, | 		const QString &path, | ||||||
| 		const Settings &settings); | 		const Settings &settings); | ||||||
|  | 	void reconfigure(const Settings &settings); | ||||||
| 
 | 
 | ||||||
| 	void open(EncryptionKey &&key, FnMut<void(Error)> &&done); | 	void open(EncryptionKey &&key, FnMut<void(Error)> &&done); | ||||||
| 	void close(FnMut<void()> &&done); | 	void close(FnMut<void()> &&done); | ||||||
|  | @ -53,9 +55,10 @@ public: | ||||||
| 		const Key &to, | 		const Key &to, | ||||||
| 		FnMut<void(Error)> &&done); | 		FnMut<void(Error)> &&done); | ||||||
| 
 | 
 | ||||||
| 	void stats(FnMut<void(Stats&&)> &&done); | 	rpl::producer<Stats> stats() const; | ||||||
| 
 | 
 | ||||||
| 	void clear(FnMut<void(Error)> &&done); | 	void clear(FnMut<void(Error)> &&done); | ||||||
|  | 	void clearByTag(uint8 tag, FnMut<void(Error)> &&done); | ||||||
| 
 | 
 | ||||||
| 	static QString BinlogFilename(); | 	static QString BinlogFilename(); | ||||||
| 	static QString CompactReadyFilename(); | 	static QString CompactReadyFilename(); | ||||||
|  | @ -101,6 +104,7 @@ private: | ||||||
| 
 | 
 | ||||||
| 	Error ioError(const QString &path) const; | 	Error ioError(const QString &path) const; | ||||||
| 
 | 
 | ||||||
|  | 	void checkSettings(); | ||||||
| 	QString computePath(Version version) const; | 	QString computePath(Version version) const; | ||||||
| 	QString binlogPath(Version version) const; | 	QString binlogPath(Version version) const; | ||||||
| 	QString binlogPath() const; | 	QString binlogPath() const; | ||||||
|  | @ -146,15 +150,24 @@ private: | ||||||
| 	uint64 countRelativeTime() const; | 	uint64 countRelativeTime() const; | ||||||
| 	EstimatedTimePoint countTimePoint() const; | 	EstimatedTimePoint countTimePoint() const; | ||||||
| 	void applyTimePoint(EstimatedTimePoint time); | 	void applyTimePoint(EstimatedTimePoint time); | ||||||
|  | 
 | ||||||
| 	uint64 pruneBeforeTime() const; | 	uint64 pruneBeforeTime() const; | ||||||
| 	void prune(); | 	void prune(); | ||||||
| 	void collectTimePrune( | 	void collectTimeStale( | ||||||
| 		base::flat_set<Key> &stale, | 		base::flat_set<Key> &stale, | ||||||
| 		int64 &staleTotalSize); | 		int64 &staleTotalSize); | ||||||
| 	void collectSizePrune( | 	void collectSizeStale( | ||||||
| 		base::flat_set<Key> &stale, | 		base::flat_set<Key> &stale, | ||||||
| 		int64 &staleTotalSize); | 		int64 &staleTotalSize); | ||||||
|  | 	void startStaleClear(); | ||||||
|  | 	void clearStaleNow(const base::flat_set<Key> &stale); | ||||||
|  | 	void clearStaleChunkDelayed(); | ||||||
|  | 	void clearStaleChunk(); | ||||||
|  | 
 | ||||||
| 	void updateStats(const Entry &was, const Entry &now); | 	void updateStats(const Entry &was, const Entry &now); | ||||||
|  | 	Stats collectStats() const; | ||||||
|  | 	void pushStatsDelayed(); | ||||||
|  | 	void pushStats(); | ||||||
| 
 | 
 | ||||||
| 	void setMapEntry(const Key &key, Entry &&entry); | 	void setMapEntry(const Key &key, Entry &&entry); | ||||||
| 	void eraseMapEntry(const Map::const_iterator &i); | 	void eraseMapEntry(const Map::const_iterator &i); | ||||||
|  | @ -197,15 +210,17 @@ private: | ||||||
| 
 | 
 | ||||||
| 	void createCleaner(); | 	void createCleaner(); | ||||||
| 	void cleanerDone(Error error); | 	void cleanerDone(Error error); | ||||||
|  | 	void clearState(); | ||||||
| 
 | 
 | ||||||
| 	crl::weak_on_queue<DatabaseObject> _weak; | 	crl::weak_on_queue<DatabaseObject> _weak; | ||||||
| 	QString _base, _path; | 	QString _base, _path; | ||||||
| 	const Settings _settings; | 	Settings _settings; | ||||||
| 	EncryptionKey _key; | 	EncryptionKey _key; | ||||||
| 	File _binlog; | 	File _binlog; | ||||||
| 	Map _map; | 	Map _map; | ||||||
| 	std::set<Key> _removing; | 	std::set<Key> _removing; | ||||||
| 	std::set<Key> _accessed; | 	std::set<Key> _accessed; | ||||||
|  | 	std::vector<Key> _stale; | ||||||
| 
 | 
 | ||||||
| 	EstimatedTimePoint _time; | 	EstimatedTimePoint _time; | ||||||
| 
 | 
 | ||||||
|  | @ -214,7 +229,10 @@ private: | ||||||
| 	uint64 _minimalEntryTime = 0; | 	uint64 _minimalEntryTime = 0; | ||||||
| 	size_type _entriesWithMinimalTimeCount = 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 _writeBundlesTimer; | ||||||
| 	base::ConcurrentTimer _pruneTimer; | 	base::ConcurrentTimer _pruneTimer; | ||||||
|  |  | ||||||
|  | @ -56,6 +56,7 @@ struct Settings { | ||||||
| 	size_type readBlockSize = 8 * 1024 * 1024; | 	size_type readBlockSize = 8 * 1024 * 1024; | ||||||
| 	size_type maxDataSize = 10 * 1024 * 1024; | 	size_type maxDataSize = 10 * 1024 * 1024; | ||||||
| 	crl::time_type writeBundleDelay = 15 * 60 * crl::time_type(1000); | 	crl::time_type writeBundleDelay = 15 * 60 * crl::time_type(1000); | ||||||
|  | 	size_type staleRemoveChunk = 256; | ||||||
| 
 | 
 | ||||||
| 	int64 compactAfterExcess = 8 * 1024 * 1024; | 	int64 compactAfterExcess = 8 * 1024 * 1024; | ||||||
| 	int64 compactAfterFullSize = 0; | 	int64 compactAfterFullSize = 0; | ||||||
|  | @ -66,6 +67,8 @@ struct Settings { | ||||||
| 	size_type totalTimeLimit = 30 * 86400; // One month in seconds.
 | 	size_type totalTimeLimit = 30 * 86400; // One month in seconds.
 | ||||||
| 	crl::time_type pruneTimeout = 5 * crl::time_type(1000); | 	crl::time_type pruneTimeout = 5 * crl::time_type(1000); | ||||||
| 	crl::time_type maxPruneCheckTimeout = 3600 * crl::time_type(1000); | 	crl::time_type maxPruneCheckTimeout = 3600 * crl::time_type(1000); | ||||||
|  | 
 | ||||||
|  | 	bool clearOnWrongKey = false; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct TaggedValue { | struct TaggedValue { | ||||||
|  | @ -80,7 +83,11 @@ struct TaggedSummary { | ||||||
| 	size_type count = 0; | 	size_type count = 0; | ||||||
| 	size_type totalSize = 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; | using Version = int32; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2949,6 +2949,7 @@ QString cachePath() { | ||||||
| 
 | 
 | ||||||
| Storage::Cache::Database::Settings cacheSettings() { | Storage::Cache::Database::Settings cacheSettings() { | ||||||
| 	auto result = Storage::Cache::Database::Settings(); | 	auto result = Storage::Cache::Database::Settings(); | ||||||
|  | 	result.clearOnWrongKey = true; | ||||||
| 	return result; | 	return result; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -2958,7 +2959,6 @@ Storage::EncryptionKey cacheKey() { | ||||||
| 	return Storage::EncryptionKey(bytes::make_vector(LocalKey->data())); | 	return Storage::EncryptionKey(bytes::make_vector(LocalKey->data())); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| class CountWaveformTask : public Task { | class CountWaveformTask : public Task { | ||||||
| public: | public: | ||||||
| 	CountWaveformTask(DocumentData *doc) | 	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 { | class RoundButton : public RippleButton, private base::Subscriber { | ||||||
| public: | 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); | 	void setText(Fn<QString()> textFactory); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -50,6 +50,8 @@ | ||||||
|       '<(submodules_loc)/xxHash', |       '<(submodules_loc)/xxHash', | ||||||
|     ], |     ], | ||||||
|     'sources': [ |     'sources': [ | ||||||
|  |       '<(src_loc)/storage/storage_databases.cpp', | ||||||
|  |       '<(src_loc)/storage/storage_databases.h', | ||||||
|       '<(src_loc)/storage/storage_encryption.cpp', |       '<(src_loc)/storage/storage_encryption.cpp', | ||||||
|       '<(src_loc)/storage/storage_encryption.h', |       '<(src_loc)/storage/storage_encryption.h', | ||||||
|       '<(src_loc)/storage/storage_encrypted_file.cpp', |       '<(src_loc)/storage/storage_encrypted_file.cpp', | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue