2013-04-15 12:48:12 +02:00
|
|
|
/******************************************************************************
|
|
|
|
* Icinga 2 *
|
2018-10-18 09:27:04 +02:00
|
|
|
* Copyright (C) 2012-2018 Icinga Development Team (https://icinga.com/) *
|
2013-04-15 12:48:12 +02:00
|
|
|
* *
|
|
|
|
* This program is free software; you can redistribute it and/or *
|
|
|
|
* modify it under the terms of the GNU General Public License *
|
|
|
|
* as published by the Free Software Foundation; either version 2 *
|
|
|
|
* of the License, or (at your option) any later version. *
|
|
|
|
* *
|
|
|
|
* This program is distributed in the hope that it will be useful, *
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
|
|
* GNU General Public License for more details. *
|
|
|
|
* *
|
|
|
|
* You should have received a copy of the GNU General Public License *
|
|
|
|
* along with this program; if not, write to the Free Software Foundation *
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
|
|
******************************************************************************/
|
|
|
|
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "icinga/legacytimeperiod.hpp"
|
2015-01-21 08:47:45 +01:00
|
|
|
#include "base/function.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/convert.hpp"
|
|
|
|
#include "base/exception.hpp"
|
|
|
|
#include "base/objectlock.hpp"
|
2014-10-19 14:21:12 +02:00
|
|
|
#include "base/logger.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/debug.hpp"
|
|
|
|
#include "base/utility.hpp"
|
2013-04-15 12:48:12 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2018-08-07 13:55:41 +02:00
|
|
|
REGISTER_FUNCTION_NONCONST(Internal, LegacyTimePeriod, &LegacyTimePeriod::ScriptFunc, "tp:begin:end");
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2013-04-17 14:26:04 +02:00
|
|
|
bool LegacyTimePeriod::IsInTimeRange(tm *begin, tm *end, int stride, tm *reference)
|
|
|
|
{
|
|
|
|
time_t tsbegin, tsend, tsref;
|
|
|
|
tsbegin = mktime(begin);
|
|
|
|
tsend = mktime(end);
|
|
|
|
tsref = mktime(reference);
|
|
|
|
|
|
|
|
if (tsref < tsbegin || tsref > tsend)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
int daynumber = (tsref - tsbegin) / (24 * 60 * 60);
|
|
|
|
|
2013-06-24 15:01:23 +02:00
|
|
|
if (stride > 1 && daynumber % stride == 0)
|
2013-04-17 14:26:04 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LegacyTimePeriod::FindNthWeekday(int wday, int n, tm *reference)
|
|
|
|
{
|
2013-05-08 12:48:45 +02:00
|
|
|
int dir, seen = 0;
|
|
|
|
|
|
|
|
if (n > 0) {
|
|
|
|
dir = 1;
|
|
|
|
} else {
|
|
|
|
n *= -1;
|
|
|
|
dir = -1;
|
|
|
|
|
|
|
|
/* Negative days are relative to the next month. */
|
|
|
|
reference->tm_mon++;
|
|
|
|
}
|
2013-04-17 14:26:04 +02:00
|
|
|
|
|
|
|
ASSERT(n > 0);
|
|
|
|
|
|
|
|
reference->tm_mday = 1;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
mktime(reference);
|
|
|
|
|
|
|
|
if (reference->tm_wday == wday) {
|
|
|
|
seen++;
|
|
|
|
|
|
|
|
if (seen == n)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-05-08 12:48:45 +02:00
|
|
|
reference->tm_mday += dir;
|
2013-04-17 14:26:04 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int LegacyTimePeriod::WeekdayFromString(const String& daydef)
|
|
|
|
{
|
|
|
|
if (daydef == "sunday")
|
|
|
|
return 0;
|
|
|
|
else if (daydef == "monday")
|
|
|
|
return 1;
|
|
|
|
else if (daydef == "tuesday")
|
|
|
|
return 2;
|
|
|
|
else if (daydef == "wednesday")
|
|
|
|
return 3;
|
|
|
|
else if (daydef == "thursday")
|
|
|
|
return 4;
|
|
|
|
else if (daydef == "friday")
|
|
|
|
return 5;
|
|
|
|
else if (daydef == "saturday")
|
|
|
|
return 6;
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int LegacyTimePeriod::MonthFromString(const String& monthdef)
|
|
|
|
{
|
|
|
|
if (monthdef == "january")
|
|
|
|
return 0;
|
|
|
|
else if (monthdef == "february")
|
|
|
|
return 1;
|
|
|
|
else if (monthdef == "march")
|
|
|
|
return 2;
|
|
|
|
else if (monthdef == "april")
|
|
|
|
return 3;
|
|
|
|
else if (monthdef == "may")
|
|
|
|
return 4;
|
|
|
|
else if (monthdef == "june")
|
|
|
|
return 5;
|
|
|
|
else if (monthdef == "july")
|
|
|
|
return 6;
|
|
|
|
else if (monthdef == "august")
|
|
|
|
return 7;
|
|
|
|
else if (monthdef == "september")
|
|
|
|
return 8;
|
|
|
|
else if (monthdef == "october")
|
|
|
|
return 9;
|
|
|
|
else if (monthdef == "november")
|
|
|
|
return 10;
|
|
|
|
else if (monthdef == "december")
|
|
|
|
return 11;
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LegacyTimePeriod::ParseTimeSpec(const String& timespec, tm *begin, tm *end, tm *reference)
|
|
|
|
{
|
2013-05-03 12:30:57 +02:00
|
|
|
/* Let mktime() figure out whether we're in DST or not. */
|
|
|
|
reference->tm_isdst = -1;
|
|
|
|
|
2013-04-17 14:26:04 +02:00
|
|
|
/* YYYY-MM-DD */
|
|
|
|
if (timespec.GetLength() == 10 && timespec[4] == '-' && timespec[7] == '-') {
|
|
|
|
int year = Convert::ToLong(timespec.SubStr(0, 4));
|
|
|
|
int month = Convert::ToLong(timespec.SubStr(5, 2));
|
2013-11-13 14:56:31 +01:00
|
|
|
int day = Convert::ToLong(timespec.SubStr(8, 2));
|
2013-04-17 14:26:04 +02:00
|
|
|
|
2016-02-09 19:04:37 +01:00
|
|
|
if (month < 1 || month > 12)
|
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid month in time specification: " + timespec));
|
|
|
|
if (day < 1 || day > 31)
|
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid day in time specification: " + timespec));
|
|
|
|
|
2013-04-17 14:26:04 +02:00
|
|
|
if (begin) {
|
2016-05-11 09:23:55 +02:00
|
|
|
*begin = *reference;
|
2013-04-17 14:26:04 +02:00
|
|
|
begin->tm_year = year - 1900;
|
2016-02-09 19:04:37 +01:00
|
|
|
begin->tm_mon = month - 1;
|
|
|
|
begin->tm_mday = day;
|
2013-04-17 14:26:04 +02:00
|
|
|
begin->tm_hour = 0;
|
|
|
|
begin->tm_min = 0;
|
|
|
|
begin->tm_sec = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (end) {
|
2016-05-11 09:23:55 +02:00
|
|
|
*end = *reference;
|
2013-04-17 14:26:04 +02:00
|
|
|
end->tm_year = year - 1900;
|
2016-02-09 19:04:37 +01:00
|
|
|
end->tm_mon = month - 1;
|
|
|
|
end->tm_mday = day;
|
2013-04-17 14:26:04 +02:00
|
|
|
end->tm_hour = 24;
|
|
|
|
end->tm_min = 0;
|
|
|
|
end->tm_sec = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-04 18:24:45 +01:00
|
|
|
std::vector<String> tokens = timespec.Split(" ");
|
2013-04-17 14:26:04 +02:00
|
|
|
|
2013-05-03 12:44:27 +02:00
|
|
|
int mon = -1;
|
2013-05-03 12:31:11 +02:00
|
|
|
|
|
|
|
if (tokens.size() > 1 && (tokens[0] == "day" || (mon = MonthFromString(tokens[0])) != -1)) {
|
|
|
|
if (mon == -1)
|
|
|
|
mon = reference->tm_mon;
|
|
|
|
|
2013-04-17 14:26:04 +02:00
|
|
|
int mday = Convert::ToLong(tokens[1]);
|
|
|
|
|
|
|
|
if (begin) {
|
|
|
|
*begin = *reference;
|
2013-05-03 12:31:11 +02:00
|
|
|
begin->tm_mon = mon;
|
2013-04-17 14:26:04 +02:00
|
|
|
begin->tm_mday = mday;
|
|
|
|
begin->tm_hour = 0;
|
|
|
|
begin->tm_min = 0;
|
|
|
|
begin->tm_sec = 0;
|
|
|
|
|
|
|
|
/* Negative days are relative to the next month. */
|
|
|
|
if (mday < 0) {
|
2016-05-11 09:23:55 +02:00
|
|
|
begin->tm_mday = mday * -1 - 1;
|
2013-04-17 14:26:04 +02:00
|
|
|
begin->tm_mon++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (end) {
|
|
|
|
*end = *reference;
|
2013-05-03 12:31:11 +02:00
|
|
|
end->tm_mon = mon;
|
2013-04-17 14:26:04 +02:00
|
|
|
end->tm_mday = mday;
|
|
|
|
end->tm_hour = 24;
|
|
|
|
end->tm_min = 0;
|
|
|
|
end->tm_sec = 0;
|
|
|
|
|
|
|
|
/* Negative days are relative to the next month. */
|
|
|
|
if (mday < 0) {
|
2016-05-11 09:23:55 +02:00
|
|
|
end->tm_mday = mday * -1 - 1;
|
2013-04-17 14:26:04 +02:00
|
|
|
end->tm_mon++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int wday;
|
|
|
|
|
2013-04-18 08:20:16 +02:00
|
|
|
if (tokens.size() >= 1 && (wday = WeekdayFromString(tokens[0])) != -1) {
|
2013-04-17 14:26:04 +02:00
|
|
|
tm myref = *reference;
|
|
|
|
|
|
|
|
if (tokens.size() > 2) {
|
2013-05-03 12:31:11 +02:00
|
|
|
mon = MonthFromString(tokens[2]);
|
2013-04-17 14:26:04 +02:00
|
|
|
|
|
|
|
if (mon == -1)
|
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid month in time specification: " + timespec));
|
|
|
|
|
|
|
|
myref.tm_mon = mon;
|
|
|
|
}
|
|
|
|
|
2013-05-03 12:44:27 +02:00
|
|
|
int n = 0;
|
2013-04-18 08:20:16 +02:00
|
|
|
|
|
|
|
if (tokens.size() > 1)
|
|
|
|
n = Convert::ToLong(tokens[1]);
|
2013-04-17 14:26:04 +02:00
|
|
|
|
|
|
|
if (begin) {
|
|
|
|
*begin = myref;
|
2013-04-18 08:20:16 +02:00
|
|
|
|
|
|
|
if (tokens.size() > 1)
|
|
|
|
FindNthWeekday(wday, n, begin);
|
|
|
|
else
|
2014-12-20 22:04:24 +01:00
|
|
|
begin->tm_mday += (7 - begin->tm_wday + wday) % 7;
|
2013-04-18 08:20:16 +02:00
|
|
|
|
2013-04-17 14:26:04 +02:00
|
|
|
begin->tm_hour = 0;
|
|
|
|
begin->tm_min = 0;
|
|
|
|
begin->tm_sec = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (end) {
|
|
|
|
*end = myref;
|
2013-04-18 08:20:16 +02:00
|
|
|
|
|
|
|
if (tokens.size() > 1)
|
|
|
|
FindNthWeekday(wday, n, end);
|
|
|
|
else
|
2014-12-20 22:04:24 +01:00
|
|
|
end->tm_mday += (7 - end->tm_wday + wday) % 7;
|
2013-04-18 08:20:16 +02:00
|
|
|
|
2013-04-17 14:26:04 +02:00
|
|
|
end->tm_hour = 0;
|
|
|
|
end->tm_min = 0;
|
|
|
|
end->tm_sec = 0;
|
|
|
|
end->tm_mday++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid time specification: " + timespec));
|
|
|
|
}
|
|
|
|
|
|
|
|
void LegacyTimePeriod::ParseTimeRange(const String& timerange, tm *begin, tm *end, int *stride, tm *reference)
|
|
|
|
{
|
|
|
|
String def = timerange;
|
|
|
|
|
|
|
|
/* Figure out the stride. */
|
|
|
|
size_t pos = def.FindFirstOf('/');
|
|
|
|
|
|
|
|
if (pos != String::NPos) {
|
2015-08-27 18:06:20 +02:00
|
|
|
String strStride = def.SubStr(pos + 1).Trim();
|
2013-04-17 14:26:04 +02:00
|
|
|
*stride = Convert::ToLong(strStride);
|
|
|
|
|
|
|
|
/* Remove the stride parameter from the definition. */
|
|
|
|
def = def.SubStr(0, pos);
|
|
|
|
} else {
|
|
|
|
*stride = 1; /* User didn't specify anything, assume default. */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Figure out whether the user has specified two dates. */
|
|
|
|
pos = def.Find("- ");
|
|
|
|
|
|
|
|
if (pos != String::NPos) {
|
2015-08-27 18:06:20 +02:00
|
|
|
String first = def.SubStr(0, pos).Trim();
|
2013-04-17 14:26:04 +02:00
|
|
|
|
2015-08-27 18:06:20 +02:00
|
|
|
String second = def.SubStr(pos + 1).Trim();
|
2013-04-17 14:26:04 +02:00
|
|
|
|
2017-12-14 15:37:20 +01:00
|
|
|
ParseTimeSpec(first, begin, nullptr, reference);
|
2013-04-17 14:26:04 +02:00
|
|
|
|
|
|
|
/* If the second definition starts with a number we need
|
|
|
|
* to add the first word from the first definition, e.g.:
|
|
|
|
* day 1 - 15 --> "day 15" */
|
|
|
|
bool is_number = true;
|
|
|
|
size_t xpos = second.FindFirstOf(' ');
|
|
|
|
String fword = second.SubStr(0, xpos);
|
|
|
|
|
|
|
|
try {
|
|
|
|
Convert::ToLong(fword);
|
|
|
|
} catch (...) {
|
|
|
|
is_number = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_number) {
|
|
|
|
xpos = first.FindFirstOf(' ');
|
|
|
|
ASSERT(xpos != String::NPos);
|
|
|
|
second = first.SubStr(0, xpos + 1) + second;
|
|
|
|
}
|
|
|
|
|
2017-12-14 15:37:20 +01:00
|
|
|
ParseTimeSpec(second, nullptr, end, reference);
|
2013-04-17 14:26:04 +02:00
|
|
|
} else {
|
2015-03-29 08:13:11 +02:00
|
|
|
ParseTimeSpec(def, begin, end, reference);
|
2013-04-17 14:26:04 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-15 12:48:12 +02:00
|
|
|
bool LegacyTimePeriod::IsInDayDefinition(const String& daydef, tm *reference)
|
|
|
|
{
|
2013-04-17 14:26:04 +02:00
|
|
|
tm begin, end;
|
|
|
|
int stride;
|
|
|
|
|
|
|
|
ParseTimeRange(daydef, &begin, &end, &stride, reference);
|
|
|
|
|
2014-10-19 17:52:17 +02:00
|
|
|
Log(LogDebug, "LegacyTimePeriod")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "ParseTimeRange: '" << daydef << "' => " << mktime(&begin)
|
|
|
|
<< " -> " << mktime(&end) << ", stride: " << stride;
|
2013-06-24 15:01:23 +02:00
|
|
|
|
2013-04-17 14:26:04 +02:00
|
|
|
return IsInTimeRange(&begin, &end, stride, reference);
|
2013-04-15 12:48:12 +02:00
|
|
|
}
|
|
|
|
|
2013-08-09 11:07:55 +02:00
|
|
|
void LegacyTimePeriod::ProcessTimeRangeRaw(const String& timerange, tm *reference, tm *begin, tm *end)
|
2013-04-15 12:48:12 +02:00
|
|
|
{
|
2018-01-04 18:24:45 +01:00
|
|
|
std::vector<String> times = timerange.Split("-");
|
2013-04-15 12:48:12 +02:00
|
|
|
|
|
|
|
if (times.size() != 2)
|
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid timerange: " + timerange));
|
|
|
|
|
2018-01-04 18:24:45 +01:00
|
|
|
std::vector<String> hd1 = times[0].Split(":");
|
2013-04-15 12:48:12 +02:00
|
|
|
|
|
|
|
if (hd1.size() != 2)
|
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid time specification: " + times[0]));
|
|
|
|
|
2018-01-04 18:24:45 +01:00
|
|
|
std::vector<String> hd2 = times[1].Split(":");
|
2013-04-15 12:48:12 +02:00
|
|
|
|
|
|
|
if (hd2.size() != 2)
|
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid time specification: " + times[1]));
|
|
|
|
|
2013-08-09 11:07:55 +02:00
|
|
|
*begin = *reference;
|
|
|
|
begin->tm_sec = 0;
|
|
|
|
begin->tm_min = Convert::ToLong(hd1[1]);
|
|
|
|
begin->tm_hour = Convert::ToLong(hd1[0]);
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2013-08-09 11:07:55 +02:00
|
|
|
*end = *reference;
|
|
|
|
end->tm_sec = 0;
|
|
|
|
end->tm_min = Convert::ToLong(hd2[1]);
|
|
|
|
end->tm_hour = Convert::ToLong(hd2[0]);
|
2015-03-29 08:05:01 +02:00
|
|
|
|
|
|
|
if (begin->tm_hour * 3600 + begin->tm_min * 60 + begin->tm_sec >=
|
2017-12-19 15:50:05 +01:00
|
|
|
end->tm_hour * 3600 + end->tm_min * 60 + end->tm_sec)
|
2015-03-29 08:05:01 +02:00
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Time period segment ends before it begins"));
|
2013-08-09 11:07:55 +02:00
|
|
|
}
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2013-08-09 11:07:55 +02:00
|
|
|
Dictionary::Ptr LegacyTimePeriod::ProcessTimeRange(const String& timestamp, tm *reference)
|
|
|
|
{
|
|
|
|
tm begin, end;
|
2015-02-12 09:12:55 +01:00
|
|
|
|
2015-03-29 08:13:11 +02:00
|
|
|
ProcessTimeRangeRaw(timestamp, reference, &begin, &end);
|
2015-02-12 09:12:55 +01:00
|
|
|
|
2018-01-11 11:17:38 +01:00
|
|
|
return new Dictionary({
|
|
|
|
{ "begin", (long)mktime(&begin) },
|
|
|
|
{ "end", (long)mktime(&end) }
|
|
|
|
});
|
2013-04-15 12:48:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void LegacyTimePeriod::ProcessTimeRanges(const String& timeranges, tm *reference, const Array::Ptr& result)
|
|
|
|
{
|
2018-01-04 18:24:45 +01:00
|
|
|
std::vector<String> ranges = timeranges.Split(",");
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const String& range : ranges) {
|
2015-03-29 08:13:11 +02:00
|
|
|
Dictionary::Ptr segment = ProcessTimeRange(range, reference);
|
2015-02-12 09:12:55 +01:00
|
|
|
|
2015-01-29 14:14:56 +01:00
|
|
|
if (segment->Get("begin") >= segment->Get("end"))
|
2015-03-29 08:05:01 +02:00
|
|
|
continue;
|
2015-01-29 14:14:56 +01:00
|
|
|
|
2013-04-15 12:48:12 +02:00
|
|
|
result->Add(segment);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-18 18:42:21 +02:00
|
|
|
Dictionary::Ptr LegacyTimePeriod::FindRunningSegment(const String& daydef, const String& timeranges, tm *reference)
|
|
|
|
{
|
|
|
|
tm begin, end, iter;
|
|
|
|
time_t tsend, tsiter, tsref;
|
|
|
|
int stride;
|
|
|
|
|
|
|
|
tsref = mktime(reference);
|
|
|
|
|
|
|
|
ParseTimeRange(daydef, &begin, &end, &stride, reference);
|
|
|
|
|
|
|
|
iter = begin;
|
|
|
|
|
|
|
|
tsend = mktime(&end);
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (IsInTimeRange(&begin, &end, stride, &iter)) {
|
|
|
|
Array::Ptr segments = new Array();
|
|
|
|
ProcessTimeRanges(timeranges, &iter, segments);
|
|
|
|
|
|
|
|
Dictionary::Ptr bestSegment;
|
|
|
|
double bestEnd;
|
|
|
|
|
|
|
|
ObjectLock olock(segments);
|
|
|
|
for (const Dictionary::Ptr& segment : segments) {
|
|
|
|
double begin = segment->Get("begin");
|
|
|
|
double end = segment->Get("end");
|
|
|
|
|
|
|
|
if (begin >= tsref || end < tsref)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!bestSegment || end > bestEnd) {
|
|
|
|
bestSegment = segment;
|
|
|
|
bestEnd = end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bestSegment)
|
|
|
|
return bestSegment;
|
|
|
|
}
|
|
|
|
|
|
|
|
iter.tm_mday++;
|
|
|
|
iter.tm_hour = 0;
|
|
|
|
iter.tm_min = 0;
|
|
|
|
iter.tm_sec = 0;
|
|
|
|
tsiter = mktime(&iter);
|
|
|
|
} while (tsiter < tsend);
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2013-11-13 14:56:31 +01:00
|
|
|
Dictionary::Ptr LegacyTimePeriod::FindNextSegment(const String& daydef, const String& timeranges, tm *reference)
|
|
|
|
{
|
|
|
|
tm begin, end, iter, ref;
|
|
|
|
time_t tsend, tsiter, tsref;
|
|
|
|
int stride;
|
|
|
|
|
|
|
|
for (int pass = 1; pass <= 2; pass++) {
|
|
|
|
if (pass == 1) {
|
|
|
|
ref = *reference;
|
|
|
|
} else {
|
|
|
|
ref = end;
|
|
|
|
ref.tm_mday++;
|
|
|
|
}
|
|
|
|
|
|
|
|
tsref = mktime(&ref);
|
|
|
|
|
|
|
|
ParseTimeRange(daydef, &begin, &end, &stride, &ref);
|
|
|
|
|
|
|
|
iter = begin;
|
|
|
|
|
|
|
|
tsend = mktime(&end);
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (IsInTimeRange(&begin, &end, stride, &iter)) {
|
2014-11-08 21:17:16 +01:00
|
|
|
Array::Ptr segments = new Array();
|
2013-11-13 14:56:31 +01:00
|
|
|
ProcessTimeRanges(timeranges, &iter, segments);
|
|
|
|
|
|
|
|
Dictionary::Ptr bestSegment;
|
|
|
|
double bestBegin;
|
|
|
|
|
2014-04-06 08:09:49 +02:00
|
|
|
ObjectLock olock(segments);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Dictionary::Ptr& segment : segments) {
|
2013-11-13 14:56:31 +01:00
|
|
|
double begin = segment->Get("begin");
|
|
|
|
|
|
|
|
if (begin < tsref)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!bestSegment || begin < bestBegin) {
|
|
|
|
bestSegment = segment;
|
|
|
|
bestBegin = begin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bestSegment)
|
|
|
|
return bestSegment;
|
|
|
|
}
|
|
|
|
|
|
|
|
iter.tm_mday++;
|
|
|
|
iter.tm_hour = 0;
|
|
|
|
iter.tm_min = 0;
|
|
|
|
iter.tm_sec = 0;
|
|
|
|
tsiter = mktime(&iter);
|
|
|
|
} while (tsiter < tsend);
|
|
|
|
}
|
|
|
|
|
2017-11-30 08:36:35 +01:00
|
|
|
return nullptr;
|
2013-11-13 14:56:31 +01:00
|
|
|
}
|
|
|
|
|
2013-04-15 12:48:12 +02:00
|
|
|
Array::Ptr LegacyTimePeriod::ScriptFunc(const TimePeriod::Ptr& tp, double begin, double end)
|
|
|
|
{
|
2014-11-08 21:17:16 +01:00
|
|
|
Array::Ptr segments = new Array();
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
Dictionary::Ptr ranges = tp->GetRanges();
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2013-04-16 10:55:23 +02:00
|
|
|
if (ranges) {
|
|
|
|
for (int i = 0; i <= (end - begin) / (24 * 60 * 60); i++) {
|
|
|
|
time_t refts = begin + i * 24 * 60 * 60;
|
2013-11-13 14:56:31 +01:00
|
|
|
tm reference = Utility::LocalTime(refts);
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2014-12-19 12:19:28 +01:00
|
|
|
#ifdef I2_DEBUG
|
2014-10-19 17:52:17 +02:00
|
|
|
Log(LogDebug, "LegacyTimePeriod")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Checking reference time " << refts;
|
2014-12-19 12:19:28 +01:00
|
|
|
#endif /* I2_DEBUG */
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2013-04-16 10:12:53 +02:00
|
|
|
ObjectLock olock(ranges);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Dictionary::Pair& kv : ranges) {
|
2013-11-30 17:42:50 +01:00
|
|
|
if (!IsInDayDefinition(kv.first, &reference)) {
|
2014-12-19 12:19:28 +01:00
|
|
|
#ifdef I2_DEBUG
|
2014-10-19 17:52:17 +02:00
|
|
|
Log(LogDebug, "LegacyTimePeriod")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Not in day definition '" << kv.first << "'.";
|
2014-12-19 12:19:28 +01:00
|
|
|
#endif /* I2_DEBUG */
|
2013-04-15 12:48:12 +02:00
|
|
|
continue;
|
2013-06-24 15:01:23 +02:00
|
|
|
}
|
|
|
|
|
2014-12-19 12:19:28 +01:00
|
|
|
#ifdef I2_DEBUG
|
2014-10-19 17:52:17 +02:00
|
|
|
Log(LogDebug, "LegacyTimePeriod")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "In day definition '" << kv.first << "'.";
|
2014-12-19 12:19:28 +01:00
|
|
|
#endif /* I2_DEBUG */
|
2013-04-15 12:48:12 +02:00
|
|
|
|
2013-11-30 17:42:50 +01:00
|
|
|
ProcessTimeRanges(kv.second, &reference, segments);
|
2013-04-15 12:48:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-19 17:52:17 +02:00
|
|
|
Log(LogDebug, "LegacyTimePeriod")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Legacy timeperiod update returned " << segments->GetLength() << " segments.";
|
2013-04-15 12:48:12 +02:00
|
|
|
|
|
|
|
return segments;
|
|
|
|
}
|