|
|
|
// Copyright 2017 The Abseil Authors.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <cctype>
|
|
|
|
#include <cstdint>
|
|
|
|
|
|
|
|
#include "absl/strings/match.h"
|
|
|
|
#include "absl/strings/string_view.h"
|
|
|
|
#include "absl/time/internal/cctz/include/cctz/time_zone.h"
|
|
|
|
#include "absl/time/time.h"
|
|
|
|
|
|
|
|
namespace cctz = absl::time_internal::cctz;
|
|
|
|
|
|
|
|
namespace absl {
|
|
|
|
ABSL_NAMESPACE_BEGIN
|
|
|
|
|
Export of internal Abseil changes
--
b2b94b9f533e4f9ae2a2df9de56ccb3b18f31d0d by Gennadiy Rozental <rogeeff@google.com>:
Internal change
PiperOrigin-RevId: 314366138
--
7a5ac6be82741aec5b327d7b67efd14d69deed6a by Gennadiy Rozental <rogeeff@google.com>:
Introduce Abseil prefixed dynamic annotation macros.
PiperOrigin-RevId: 314228914
--
33e7c605cb1be9bd48bd3590b1eedccd68d3bd1b by Gennadiy Rozental <rogeeff@google.com>:
Fixing missing includes and remove unnecessary ones.
PiperOrigin-RevId: 314217909
--
6a3c5c26bfa13317bf0a880f13d0e4be0b971b76 by Gennadiy Rozental <rogeeff@google.com>:
Import of CCTZ from GitHub.
PiperOrigin-RevId: 314209576
--
f7795aa68020af4a6b4905531ba951e04b88966a by Abseil Team <absl-team@google.com>:
absl::FormatTime() and absl::ParseTime() format specifiers:
- %EZ now accepts 'z' in addition to 'Z' as a synonym for +00:00.
- %ET is introduced, producing 'T' on output, and accepting 'T'
or 't' on input. This is for the RFC3339 date-time separator.
PiperOrigin-RevId: 313945137
--
87c437ce3aab3f59a7546e44a28cd1c8aaa152c3 by Laramie Leavitt <lar@google.com>:
Rollback
PiperOrigin-RevId: 313868206
--
8049b74349486a0026932b86d29c380b195e1cba by Laramie Leavitt <lar@google.com>:
Remove the MockingBitGenBase base class in favor of type-erasure in BitGenRef.
In Abseil random, mocking was split across two different classes,
MockingBitGenBase and MockingBitGen. This split existed because Google Mock is a
test-only library that we don't link into production, so MockingBitGenBase
provided a low-overhead scaffold used to lookup mocks when in test code, but
which is unused in production code.
That has been replaced by type-erasure which looks for a method named
CallImpl with the correct signature.
Weaken the coupling between MockingBitGen, DistributionCaller, and MockOverloadSet.
Rename CallImpl to InvokeMock()
Previously, the implementation of DistributionCaller was also split across different files using explicit instantiation of the DistributionCaller struct and some details in the Mocking classes. Now Distribution caller uses the presence of the InvokeMock() method to choose whether to use the mockable call path or the default call path.
PiperOrigin-RevId: 313848695
--
1741d80e08050e1939605f70ca6ff64809785c85 by Gennadiy Rozental <rogeeff@google.com>:
Introduce public interface to access reflection handle corresponding the flag.
This interface will be the only official way to access flag reflection information from the flag object. The other internal methods will eventually disappear.
PiperOrigin-RevId: 313734006
--
c375bead457de29d9c29595d16c66d3e5125b585 by Abseil Team <absl-team@google.com>:
Improve documentation of absl::c_partial_sort_copy.
This function takes no middle parameter, instead using the size of the result
container to determine the size of the partial sort.
PiperOrigin-RevId: 313656062
--
bbc759d43656b1b996ad558f23c852a9f14129d2 by Gennadiy Rozental <rogeeff@google.com>:
Eliminate dynamic annotation symbols in the library.
PiperOrigin-RevId: 313650817
GitOrigin-RevId: b2b94b9f533e4f9ae2a2df9de56ccb3b18f31d0d
Change-Id: Ic7a11bbcb723f3ff6a7e2f214bff0a92c6f8ab4d
5 years ago
|
|
|
ABSL_DLL extern const char RFC3339_full[] = "%Y-%m-%d%ET%H:%M:%E*S%Ez";
|
|
|
|
ABSL_DLL extern const char RFC3339_sec[] = "%Y-%m-%d%ET%H:%M:%S%Ez";
|
|
|
|
|
|
|
|
ABSL_DLL extern const char RFC1123_full[] = "%a, %d %b %E4Y %H:%M:%S %z";
|
|
|
|
ABSL_DLL extern const char RFC1123_no_wday[] = "%d %b %E4Y %H:%M:%S %z";
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
const char kInfiniteFutureStr[] = "infinite-future";
|
|
|
|
const char kInfinitePastStr[] = "infinite-past";
|
|
|
|
|
|
|
|
struct cctz_parts {
|
|
|
|
cctz::time_point<cctz::seconds> sec;
|
|
|
|
cctz::detail::femtoseconds fem;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline cctz::time_point<cctz::seconds> unix_epoch() {
|
|
|
|
return std::chrono::time_point_cast<cctz::seconds>(
|
|
|
|
std::chrono::system_clock::from_time_t(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Splits a Time into seconds and femtoseconds, which can be used with CCTZ.
|
|
|
|
// Requires that 't' is finite. See duration.cc for details about rep_hi and
|
|
|
|
// rep_lo.
|
|
|
|
cctz_parts Split(absl::Time t) {
|
|
|
|
const auto d = time_internal::ToUnixDuration(t);
|
|
|
|
const int64_t rep_hi = time_internal::GetRepHi(d);
|
|
|
|
const int64_t rep_lo = time_internal::GetRepLo(d);
|
|
|
|
const auto sec = unix_epoch() + cctz::seconds(rep_hi);
|
|
|
|
const auto fem = cctz::detail::femtoseconds(rep_lo * (1000 * 1000 / 4));
|
|
|
|
return {sec, fem};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Joins the given seconds and femtoseconds into a Time. See duration.cc for
|
|
|
|
// details about rep_hi and rep_lo.
|
|
|
|
absl::Time Join(const cctz_parts& parts) {
|
|
|
|
const int64_t rep_hi = (parts.sec - unix_epoch()).count();
|
|
|
|
const uint32_t rep_lo =
|
|
|
|
static_cast<uint32_t>(parts.fem.count() / (1000 * 1000 / 4));
|
|
|
|
const auto d = time_internal::MakeDuration(rep_hi, rep_lo);
|
|
|
|
return time_internal::FromUnixDuration(d);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
std::string FormatTime(absl::string_view format, absl::Time t,
|
|
|
|
absl::TimeZone tz) {
|
|
|
|
if (t == absl::InfiniteFuture()) return std::string(kInfiniteFutureStr);
|
|
|
|
if (t == absl::InfinitePast()) return std::string(kInfinitePastStr);
|
|
|
|
const auto parts = Split(t);
|
|
|
|
return cctz::detail::format(std::string(format), parts.sec, parts.fem,
|
|
|
|
cctz::time_zone(tz));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string FormatTime(absl::Time t, absl::TimeZone tz) {
|
|
|
|
return FormatTime(RFC3339_full, t, tz);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string FormatTime(absl::Time t) {
|
|
|
|
return absl::FormatTime(RFC3339_full, t, absl::LocalTimeZone());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ParseTime(absl::string_view format, absl::string_view input,
|
|
|
|
absl::Time* time, std::string* err) {
|
|
|
|
return absl::ParseTime(format, input, absl::UTCTimeZone(), time, err);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the input string does not contain an explicit UTC offset, interpret
|
|
|
|
// the fields with respect to the given TimeZone.
|
|
|
|
bool ParseTime(absl::string_view format, absl::string_view input,
|
|
|
|
absl::TimeZone tz, absl::Time* time, std::string* err) {
|
|
|
|
auto strip_leading_space = [](absl::string_view* sv) {
|
|
|
|
while (!sv->empty()) {
|
|
|
|
if (!std::isspace(sv->front())) return;
|
|
|
|
sv->remove_prefix(1);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Portable toolchains means we don't get nice constexpr here.
|
|
|
|
struct Literal {
|
|
|
|
const char* name;
|
|
|
|
size_t size;
|
|
|
|
absl::Time value;
|
|
|
|
};
|
|
|
|
static Literal literals[] = {
|
|
|
|
{kInfiniteFutureStr, strlen(kInfiniteFutureStr), InfiniteFuture()},
|
|
|
|
{kInfinitePastStr, strlen(kInfinitePastStr), InfinitePast()},
|
|
|
|
};
|
|
|
|
strip_leading_space(&input);
|
|
|
|
for (const auto& lit : literals) {
|
|
|
|
if (absl::StartsWith(input, absl::string_view(lit.name, lit.size))) {
|
|
|
|
absl::string_view tail = input;
|
|
|
|
tail.remove_prefix(lit.size);
|
|
|
|
strip_leading_space(&tail);
|
|
|
|
if (tail.empty()) {
|
|
|
|
*time = lit.value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string error;
|
|
|
|
cctz_parts parts;
|
|
|
|
const bool b =
|
|
|
|
cctz::detail::parse(std::string(format), std::string(input),
|
|
|
|
cctz::time_zone(tz), &parts.sec, &parts.fem, &error);
|
|
|
|
if (b) {
|
|
|
|
*time = Join(parts);
|
|
|
|
} else if (err != nullptr) {
|
|
|
|
*err = error;
|
|
|
|
}
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Functions required to support absl::Time flags.
|
|
|
|
bool AbslParseFlag(absl::string_view text, absl::Time* t, std::string* error) {
|
|
|
|
return absl::ParseTime(RFC3339_full, text, absl::UTCTimeZone(), t, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string AbslUnparseFlag(absl::Time t) {
|
|
|
|
return absl::FormatTime(RFC3339_full, t, absl::UTCTimeZone());
|
|
|
|
}
|
|
|
|
bool ParseFlag(const std::string& text, absl::Time* t, std::string* error) {
|
|
|
|
return absl::ParseTime(RFC3339_full, text, absl::UTCTimeZone(), t, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string UnparseFlag(absl::Time t) {
|
|
|
|
return absl::FormatTime(RFC3339_full, t, absl::UTCTimeZone());
|
|
|
|
}
|
|
|
|
|
|
|
|
ABSL_NAMESPACE_END
|
|
|
|
} // namespace absl
|