tdlight/td/telegram/BusinessWorkHours.cpp

180 lines
6.9 KiB
C++
Raw Normal View History

2024-02-20 15:22:02 +03:00
//
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2024
//
// Distributed under the Boost Software License, Version 1.0. (See accompanying
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
//
#include "td/telegram/BusinessWorkHours.h"
#include "td/utils/algorithm.h"
#include "td/utils/format.h"
2024-03-08 13:03:23 +03:00
#include "td/utils/logging.h"
#include <algorithm>
2024-02-20 15:22:02 +03:00
namespace td {
2024-02-27 16:03:34 +03:00
td_api::object_ptr<td_api::businessOpeningHoursInterval>
BusinessWorkHours::WorkHoursInterval::get_business_opening_hours_interval_object() const {
return td_api::make_object<td_api::businessOpeningHoursInterval>(start_minute_, end_minute_);
2024-02-20 15:22:02 +03:00
}
2024-02-20 15:55:17 +03:00
telegram_api::object_ptr<telegram_api::businessWeeklyOpen>
BusinessWorkHours::WorkHoursInterval::get_input_business_weekly_open() const {
return telegram_api::make_object<telegram_api::businessWeeklyOpen>(start_minute_, end_minute_);
}
2024-02-20 15:22:02 +03:00
BusinessWorkHours::BusinessWorkHours(telegram_api::object_ptr<telegram_api::businessWorkHours> &&work_hours) {
if (work_hours != nullptr) {
work_hours_ = transform(work_hours->weekly_open_,
[](const telegram_api::object_ptr<telegram_api::businessWeeklyOpen> &weekly_open) {
return WorkHoursInterval(weekly_open->start_minute_, weekly_open->end_minute_);
});
2024-03-08 13:03:23 +03:00
sanitize_work_hours();
2024-02-20 15:22:02 +03:00
time_zone_id_ = std::move(work_hours->timezone_id_);
}
}
2024-02-27 16:03:34 +03:00
BusinessWorkHours::BusinessWorkHours(td_api::object_ptr<td_api::businessOpeningHours> &&work_hours) {
2024-02-20 15:55:17 +03:00
if (work_hours != nullptr) {
2024-02-27 16:03:34 +03:00
work_hours_ = transform(work_hours->opening_hours_,
[](const td_api::object_ptr<td_api::businessOpeningHoursInterval> &interval) {
return WorkHoursInterval(interval->start_minute_, interval->end_minute_);
});
2024-03-08 13:03:23 +03:00
sanitize_work_hours();
2024-02-20 15:55:17 +03:00
time_zone_id_ = std::move(work_hours->time_zone_id_);
}
}
2024-02-20 15:22:02 +03:00
bool BusinessWorkHours::is_empty() const {
return work_hours_.empty();
}
2024-02-27 16:03:34 +03:00
td_api::object_ptr<td_api::businessOpeningHours> BusinessWorkHours::get_business_opening_hours_object() const {
2024-02-20 15:22:02 +03:00
if (is_empty()) {
return nullptr;
}
vector<td_api::object_ptr<td_api::businessOpeningHoursInterval>> intervals;
for (const auto &work_hour : work_hours_) {
auto interval = work_hour;
while (interval.start_minute_ / (24 * 60) + 1 < interval.end_minute_ / (24 * 60)) {
auto prefix = interval;
prefix.end_minute_ = (interval.start_minute_ / (24 * 60) + 1) * 24 * 60;
interval.start_minute_ = prefix.end_minute_;
intervals.push_back(prefix.get_business_opening_hours_interval_object());
}
intervals.push_back(interval.get_business_opening_hours_interval_object());
}
return td_api::make_object<td_api::businessOpeningHours>(time_zone_id_, std::move(intervals));
2024-02-20 15:22:02 +03:00
}
2024-02-20 15:55:17 +03:00
telegram_api::object_ptr<telegram_api::businessWorkHours> BusinessWorkHours::get_input_business_work_hours() const {
if (is_empty()) {
return nullptr;
}
return telegram_api::make_object<telegram_api::businessWorkHours>(
0, false, time_zone_id_, transform(work_hours_, [](const WorkHoursInterval &interval) {
return interval.get_input_business_weekly_open();
}));
}
2024-03-08 13:03:23 +03:00
void BusinessWorkHours::sanitize_work_hours() {
// remove invalid work hour intervals
td::remove_if(work_hours_, [](const WorkHoursInterval &interval) {
if (interval.start_minute_ >= interval.end_minute_ || interval.start_minute_ < 0 ||
interval.end_minute_ > 8 * 24 * 60) {
LOG(INFO) << "Ignore interval " << interval;
return true;
}
return false;
});
combine_work_hour_intervals();
}
void BusinessWorkHours::combine_work_hour_intervals() {
if (work_hours_.empty()) {
return;
}
// sort intervals
std::sort(work_hours_.begin(), work_hours_.end(), [](const WorkHoursInterval &lhs, const WorkHoursInterval &rhs) {
return lhs.start_minute_ < rhs.start_minute_;
});
// combine intersecting intervals
size_t j = 0;
for (size_t i = 1; i < work_hours_.size(); i++) {
CHECK(work_hours_[i].start_minute_ >= work_hours_[j].start_minute_);
if (work_hours_[i].start_minute_ <= work_hours_[j].end_minute_) {
work_hours_[j].end_minute_ = max(work_hours_[j].end_minute_, work_hours_[i].end_minute_);
} else {
work_hours_[++j] = work_hours_[i];
}
}
work_hours_.resize(j + 1);
// there must be no intervals longer than 1 week
for (auto &interval : work_hours_) {
interval.end_minute_ = min(interval.end_minute_, interval.start_minute_ + 7 * 24 * 60);
}
CHECK(!work_hours_.empty());
// if the last interval can be exactly merged with the first one, merge them
if (work_hours_[0].start_minute_ != 0 &&
work_hours_[0].start_minute_ + 7 * 24 * 60 == work_hours_.back().end_minute_) {
if (work_hours_.back().start_minute_ >= 7 * 24 * 60) {
work_hours_[0].start_minute_ = work_hours_.back().start_minute_ - 7 * 24 * 60;
work_hours_.pop_back();
CHECK(!work_hours_.empty());
} else {
work_hours_[0].start_minute_ = 0;
work_hours_.back().end_minute_ = 7 * 24 * 60;
}
}
// if there are intervals that intersect the first interval or start after the end of the week,
// then they must be normalized
auto max_minute = work_hours_[0].start_minute_ + 7 * 24 * 60;
if (work_hours_.back().end_minute_ > max_minute || work_hours_.back().start_minute_ >= 7 * 24 * 60) {
auto size = work_hours_.size();
for (size_t i = 1; i < size; i++) {
if (work_hours_[i].start_minute_ >= 7 * 24 * 60) {
work_hours_[i].start_minute_ -= 7 * 24 * 60;
work_hours_[i].end_minute_ -= 7 * 24 * 60;
} else if (work_hours_[i].end_minute_ > max_minute) {
work_hours_.emplace_back(max_minute - 7 * 24 * 60, work_hours_[i].end_minute_ - 7 * 24 * 60);
work_hours_[i].end_minute_ = max_minute;
}
}
combine_work_hour_intervals();
}
}
2024-02-20 15:22:02 +03:00
bool operator==(const BusinessWorkHours::WorkHoursInterval &lhs, const BusinessWorkHours::WorkHoursInterval &rhs) {
return lhs.start_minute_ == rhs.start_minute_ && lhs.end_minute_ == rhs.end_minute_;
}
bool operator!=(const BusinessWorkHours::WorkHoursInterval &lhs, const BusinessWorkHours::WorkHoursInterval &rhs) {
return !(lhs == rhs);
}
StringBuilder &operator<<(StringBuilder &string_builder, const BusinessWorkHours::WorkHoursInterval &interval) {
return string_builder << '[' << interval.start_minute_ << ',' << interval.end_minute_ << ')';
}
bool operator==(const BusinessWorkHours &lhs, const BusinessWorkHours &rhs) {
return lhs.work_hours_ == rhs.work_hours_ && lhs.time_zone_id_ == rhs.time_zone_id_;
}
bool operator!=(const BusinessWorkHours &lhs, const BusinessWorkHours &rhs) {
return !(lhs == rhs);
}
StringBuilder &operator<<(StringBuilder &string_builder, const BusinessWorkHours &work_hours) {
return string_builder << "BusinessWorkHours[" << work_hours.work_hours_ << " in " << work_hours.time_zone_id_ << ']';
}
} // namespace td