20#include <winpr/config.h> 
   21#include <winpr/assert.h> 
   22#include <winpr/string.h> 
   23#include <winpr/synch.h> 
   24#include <winpr/json.h> 
   25#include <winpr/path.h> 
   26#include <winpr/file.h> 
   32#define TAG WINPR_TAG("timezone.utils") 
   34#if defined(WITH_TIMEZONE_ICU) 
   35#include <unicode/ucal.h> 
   37#include "WindowsZones.h" 
   40#include "TimeZoneNameMap.h" 
   42#if defined(WITH_TIMEZONE_COMPILED) 
   43#include "TimeZoneNameMap_static.h" 
   50} TimeZoneNameMapContext;
 
   52static TimeZoneNameMapContext tz_context = { 0 };
 
   58  free(entry->DaylightName);
 
   59  free(entry->DisplayName);
 
   62  free(entry->StandardName);
 
   74  if (entry->DaylightName)
 
   75    clone.DaylightName = _strdup(entry->DaylightName);
 
   76  if (entry->DisplayName)
 
   77    clone.DisplayName = _strdup(entry->DisplayName);
 
   79    clone.Iana = _strdup(entry->Iana);
 
   81    clone.Id = _strdup(entry->Id);
 
   82  if (entry->StandardName)
 
   83    clone.StandardName = _strdup(entry->StandardName);
 
   87static void tz_context_free(
void)
 
   89  for (
size_t x = 0; x < tz_context.count; x++)
 
   90    tz_entry_free(&tz_context.entries[x]);
 
   91  free(tz_context.entries);
 
   93  tz_context.entries = NULL;
 
   96#if defined(WITH_TIMEZONE_FROM_FILE) && defined(WITH_WINPR_JSON) 
   97static char* tz_get_object_str(WINPR_JSON* json, 
size_t pos, 
const char* name)
 
  102    WLog_WARN(TAG, 
"Invalid JSON entry at entry %" PRIuz 
", missing an Object named '%s'", pos,
 
  111              "Invalid JSON entry at entry %" PRIuz 
", Object named '%s': Not of type string",
 
  119    WLog_WARN(TAG, 
"Invalid JSON entry at entry %" PRIuz 
", Object named '%s': NULL string",
 
  132    WLog_WARN(TAG, 
"Invalid JSON entry at entry %" PRIuz 
", expected an array", pos);
 
  136  entry->Id = tz_get_object_str(json, pos, 
"Id");
 
  137  entry->StandardName = tz_get_object_str(json, pos, 
"StandardName");
 
  138  entry->DisplayName = tz_get_object_str(json, pos, 
"DisplayName");
 
  139  entry->DaylightName = tz_get_object_str(json, pos, 
"DaylightName");
 
  140  entry->Iana = tz_get_object_str(json, pos, 
"Iana");
 
  141  if (!entry->Id || !entry->StandardName || !entry->DisplayName || !entry->DaylightName ||
 
  144    tz_entry_free(entry);
 
  150static WINPR_JSON* load_timezones_from_file(
const char* filename)
 
  154    WLog_WARN(TAG, 
"Timezone resource file '%s' is not a valid JSON file", filename);
 
  159static BOOL reallocate_context(TimeZoneNameMapContext* context, 
size_t size_to_add)
 
  167                "Failed to reallocate TimeZoneNameMapEntry::entries to %" PRIuz 
" elements",
 
  168                context->count + size_to_add);
 
  171    const size_t offset = context->count;
 
  172    context->entries = tmp;
 
  173    context->count += size_to_add;
 
  180static BOOL CALLBACK load_timezones(
PINIT_ONCE once, PVOID param, PVOID* pvcontext)
 
  182  TimeZoneNameMapContext* context = param;
 
  183  WINPR_ASSERT(context);
 
  184  WINPR_UNUSED(pvcontext);
 
  187  const TimeZoneNameMapContext empty = { 0 };
 
  190#if defined(WITH_TIMEZONE_FROM_FILE) && defined(WITH_WINPR_JSON) 
  192    WINPR_JSON* json = NULL;
 
  193    char* filename = GetCombinedPath(WINPR_RESOURCE_ROOT, 
"TimeZoneNameMap.json");
 
  196      WLog_WARN(TAG, 
"Could not create WinPR timezone resource filename");
 
  200    json = load_timezones_from_file(filename);
 
  206      WLog_WARN(TAG, 
"Invalid top level JSON type in file %s, expected an array", filename);
 
  213      WLog_WARN(TAG, 
"Invalid top level JSON type in file %s, expected an array", filename);
 
  217    const size_t offset = context->count;
 
  218    if (!reallocate_context(context, count))
 
  220    for (
size_t x = 0; x < count; x++)
 
  223      if (!tz_parse_json_entry(entry, x, &context->entries[offset + x]))
 
  233#if defined(WITH_TIMEZONE_COMPILED) 
  235    const size_t offset = context->count;
 
  236    if (!reallocate_context(context, TimeZoneNameMapSize))
 
  238    for (
size_t x = 0; x < TimeZoneNameMapSize; x++)
 
  239      context->entries[offset + x] = tz_entry_clone(&TimeZoneNameMap[x]);
 
  243  (void)atexit(tz_context_free);
 
  249  static INIT_ONCE init_guard = INIT_ONCE_STATIC_INIT;
 
  251  InitOnceExecuteOnce(&init_guard, load_timezones, &tz_context, NULL);
 
  252  if (index >= tz_context.count)
 
  254  return &tz_context.entries[index];
 
  262    case TIME_ZONE_NAME_IANA:
 
  264    case TIME_ZONE_NAME_ID:
 
  266    case TIME_ZONE_NAME_STANDARD:
 
  267      return entry->StandardName;
 
  268    case TIME_ZONE_NAME_DISPLAY:
 
  269      return entry->DisplayName;
 
  270    case TIME_ZONE_NAME_DAYLIGHT:
 
  271      return entry->DaylightName;
 
  279  if (!entry || !iana || !entry->Iana)
 
  281  return strcmp(iana, entry->Iana) == 0;
 
  286  if (!entry || !
id || !entry->Id)
 
  288  return strcmp(
id, entry->Id) == 0;
 
  291static const char* get_for_type(
const char* val, TimeZoneNameType type,
 
  304      return return_type(entry, type);
 
  308#if defined(WITH_TIMEZONE_ICU) 
  309static char* get_wzid_icu(
const UChar* utzid, 
size_t utzid_len)
 
  312  UErrorCode error = U_ZERO_ERROR;
 
  314  int32_t rc = ucal_getWindowsTimeZoneID(utzid, WINPR_ASSERTING_INT_CAST(int32_t, utzid_len),
 
  316  if ((error == U_BUFFER_OVERFLOW_ERROR) && (rc > 0))
 
  319    UChar* wzid = calloc((
size_t)rc + 1, 
sizeof(UChar));
 
  322      UErrorCode error2 = U_ZERO_ERROR;
 
  323      int32_t rc2 = ucal_getWindowsTimeZoneID(
 
  324          utzid, WINPR_ASSERTING_INT_CAST(int32_t, utzid_len), wzid, rc, &error2);
 
  325      if (U_SUCCESS(error2) && (rc2 > 0))
 
  326        res = ConvertWCharNToUtf8Alloc(wzid, (
size_t)rc, NULL);
 
  333static char* get(
const char* iana)
 
  335  size_t utzid_len = 0;
 
  336  UChar* utzid = ConvertUtf8ToWCharAlloc(iana, &utzid_len);
 
  340  char* wzid = get_wzid_icu(utzid, utzid_len);
 
  345static const char* map_fallback(
const char* iana, TimeZoneNameType type)
 
  347  char* wzid = get(iana);
 
  351  const char* res = get_for_type(wzid, type, id_cmp);
 
  356static const char* map_fallback(
const char* iana, WINPR_ATTR_UNUSED TimeZoneNameType type)
 
  361  for (
size_t x = 0; x < WindowsZonesNrElements; x++)
 
  364    if (strchr(entry->tzid, 
' '))
 
  366      const char* res = NULL;
 
  367      char* tzid = _strdup(entry->tzid);
 
  371        char* space = strchr(tzid, 
' ');
 
  374        if (strcmp(tzid, iana) == 0)
 
  376          res = entry->windows;
 
  385    else if (strcmp(entry->tzid, iana) == 0)
 
  386      return entry->windows;
 
  393const char* TimeZoneIanaToWindows(
const char* iana, TimeZoneNameType type)
 
  398  const char* val = get_for_type(iana, type, iana_cmp);
 
  402  const char* wzid = map_fallback(iana, type);
 
  406  return get_for_type(wzid, type, id_cmp);
 
WINPR_API BOOL WINPR_JSON_HasObjectItem(const WINPR_JSON *object, const char *string)
Check if JSON has an object matching the name.
WINPR_API BOOL WINPR_JSON_IsString(const WINPR_JSON *item)
Check if JSON item is of type String.
WINPR_API WINPR_JSON * WINPR_JSON_ParseFromFile(const char *filename)
Parse a JSON string read from a file filename.
WINPR_API WINPR_JSON * WINPR_JSON_GetArrayItem(const WINPR_JSON *array, size_t index)
Return a pointer to an item in the array.
WINPR_API WINPR_JSON * WINPR_JSON_GetObjectItemCaseSensitive(const WINPR_JSON *object, const char *string)
Same as WINPR_JSON_GetObjectItem but with case sensitive matching.
WINPR_API BOOL WINPR_JSON_IsObject(const WINPR_JSON *item)
Check if JSON item is of type Object.
WINPR_API const char * WINPR_JSON_GetStringValue(WINPR_JSON *item)
Return the String value of a JSON item.
WINPR_API void WINPR_JSON_Delete(WINPR_JSON *item)
Delete a WinPR JSON wrapper object.
WINPR_API size_t WINPR_JSON_GetArraySize(const WINPR_JSON *array)
Get the number of arrayitems from an array.
WINPR_API BOOL WINPR_JSON_IsArray(const WINPR_JSON *item)
Check if JSON item is of type Array.