llvm/libcxx/test/libcxx/time/time.zone/time.zone.timezone/time.zone.members/get_info.sys_time.pass.cpp

//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

// UNSUPPORTED: c++03, c++11, c++14, c++17
// UNSUPPORTED: no-filesystem, no-localization, no-tzdb

// XFAIL: libcpp-has-no-experimental-tzdb
// XFAIL: availability-tzdb-missing

// <chrono>

// class time_zone;

// template <class _Duration>
//   sys_info get_info(const sys_time<_Duration>& time) const;

// tests the parts not validated in the public test
// - Validates a zone with an UNTIL in its last continuation is corrupt
// - The formatting of the FORMAT field's constrains
// - Formatting of "%z", this is valid but not present in the actual database

#include <algorithm>
#include <cassert>
#include <fstream>
#include <chrono>
#include <format>

#include "test_macros.h"
#include "assert_macros.h"
#include "concat_macros.h"
#include "filesystem_test_helper.h"
#include "test_tzdb.h"

/***** ***** HELPERS ***** *****/

scoped_test_env env;
[[maybe_unused]] const std::filesystem::path dir = env.create_dir("zoneinfo");
const std::filesystem::path file                 = env.create_file("zoneinfo/tzdata.zi");

std::string_view std::chrono::__libcpp_tzdb_directory() {
  static std::string result = dir.string();
  return result;
}

static void write(std::string_view input) {
  static int version = 0;

  std::ofstream f{file};
  f << "# version " << version++ << '\n';
  f.write(input.data(), input.size());
}

static const std::chrono::tzdb& parse(std::string_view input) {
  write(input);
  return std::chrono::reload_tzdb();
}

[[nodiscard]] static std::chrono::sys_seconds to_sys_seconds(int year) {
  std::chrono::year_month_day result{std::chrono::year{year}, std::chrono::January, std::chrono::day{1}};

  return std::chrono::time_point_cast<std::chrono::seconds>(static_cast<std::chrono::sys_days>(result));
}

static void test_exception([[maybe_unused]] std::string_view input, [[maybe_unused]] std::string_view what) {
#ifndef TEST_HAS_NO_EXCEPTIONS
  const std::chrono::tzdb& tzdb    = parse(input);
  const std::chrono::time_zone* tz = tzdb.locate_zone("Format");
  TEST_VALIDATE_EXCEPTION(
      std::runtime_error,
      [&]([[maybe_unused]] const std::runtime_error& e) {
        TEST_LIBCPP_REQUIRE(
            e.what() == what,
            TEST_WRITE_CONCATENATED("\nExpected exception ", what, "\nActual exception   ", e.what(), '\n'));
      },
      TEST_IGNORE_NODISCARD tz->get_info(to_sys_seconds(2000)));
#endif // TEST_HAS_NO_EXCEPTIONS
}

static void zone_without_until_entry() {
#ifndef TEST_HAS_NO_EXCEPTIONS
  const std::chrono::tzdb& tzdb = parse(
      R"(
Z America/Paramaribo -3:40:40 - LMT 1911
-3:40:52 - PMT 1935
-3:40:36 - PMT 1945 O
-3:30 - -0330 1984 O
# -3 - -03 Commented out so the last entry has an UNTIL field.
)");
  const std::chrono::time_zone* tz = tzdb.locate_zone("America/Paramaribo");

  TEST_IGNORE_NODISCARD tz->get_info(to_sys_seconds(1984));
  TEST_VALIDATE_EXCEPTION(
      std::runtime_error,
      [&]([[maybe_unused]] const std::runtime_error& e) {
        std::string what = "tzdb: corrupt db";
        TEST_LIBCPP_REQUIRE(
            e.what() == what,
            TEST_WRITE_CONCATENATED("\nExpected exception ", what, "\nActual exception   ", e.what(), '\n'));
      },
      TEST_IGNORE_NODISCARD tz->get_info(to_sys_seconds(1985)));
#endif // TEST_HAS_NO_EXCEPTIONS
}

static void invalid_format() {
  test_exception(
      R"(
R F 2000 max - Jan 5 0 0 foo
Z Format 0 F %zandfoo)",
      "corrupt tzdb FORMAT field: %z should be the entire contents, instead contains '%zandfoo'");

  test_exception(
      R"(
R F 2000 max - Jan 5 0 0 foo
Z Format 0 F %q)",
      "corrupt tzdb FORMAT field: invalid sequence '%q' found, expected %s or %z");

  test_exception(
      R"(
R F 2000 max - Jan 5 0 0 foo
Z Format 0 F !)",
      "corrupt tzdb FORMAT field: invalid character '!' found, expected +, -, or an alphanumeric value");

  test_exception(
      R"(
R F 2000 max - Jan 5 0 0 foo
Z Format 0 F @)",
      "corrupt tzdb FORMAT field: invalid character '@' found, expected +, -, or an alphanumeric value");

  test_exception(
      R"(
R F 2000 max - Jan 5 0 0 foo
Z Format 0 F $)",
      "corrupt tzdb FORMAT field: invalid character '$' found, expected +, -, or an alphanumeric value");

  test_exception(
      R"(
R F 1970 max - Jan 5 0 0 foo
Z Format 0 F %)",
      "corrupt tzdb FORMAT field: input ended with the start of the escape sequence '%'");

  test_exception(
      R"(
R F 2000 max - Jan 5 0 0 -
Z Format 0 F %s)",
      "corrupt tzdb FORMAT field: result is empty");
}

static void test_abbrev(std::string_view input, std::string_view expected) {
  const std::chrono::tzdb& tzdb    = parse(input);
  const std::chrono::time_zone* tz = tzdb.locate_zone("Format");
  std::string result               = tz->get_info(to_sys_seconds(2000)).abbrev;
  TEST_LIBCPP_REQUIRE(result == expected, TEST_WRITE_CONCATENATED("\nExpected ", expected, "\nActual ", result, '\n'));
}

// This format is valid, however is not used in the tzdata.zi.
static void percentage_z_format() {
  test_abbrev(
      R"(
R F 1999 max - Jan 5 0 0 foo
Z Format 0 F %z)",
      "+00");

  test_abbrev(
      R"(
R F 1999 max - Jan 5 0 1 foo
Z Format 0 F %z)",
      "+01");

  test_abbrev(
      R"(
R F 1999 max - Jan 5 0 -1 foo
Z Format 0 F %z)",
      "-01");

  test_abbrev(
      R"(
R F 1999 max - Jan 5 0 0 foo
Z Format 0:45 F %z)",
      "+0045");

  test_abbrev(
      R"(
R F 1999 max - Jan 5 0 -1 foo
Z Format 0:45 F %z)",
      "-0015");
}

int main(int, const char**) {
  zone_without_until_entry();
  invalid_format();
  percentage_z_format();

  return 0;
}