// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "google_apis/drive/time_util.h" #include <string> #include <vector> #include "base/strings/string_number_conversions.h" #include "base/strings/string_util.h" #include "base/strings/stringprintf.h" #include "base/time/time.h" namespace google_apis { namespace util { namespace { const char kNullTimeString[] = "null"; bool ParseTimezone(const base::StringPiece& timezone, bool ahead, int* out_offset_to_utc_in_minutes) { DCHECK(out_offset_to_utc_in_minutes); std::vector<base::StringPiece> parts; int num_of_token = Tokenize(timezone, ":", &parts); int hour = 0; if (!base::StringToInt(parts[0], &hour)) return false; int minute = 0; if (num_of_token > 1 && !base::StringToInt(parts[1], &minute)) return false; *out_offset_to_utc_in_minutes = (hour * 60 + minute) * (ahead ? +1 : -1); return true; } } // namespace bool GetTimeFromString(const base::StringPiece& raw_value, base::Time* parsed_time) { base::StringPiece date; base::StringPiece time_and_tz; base::StringPiece time; base::Time::Exploded exploded = {0}; bool has_timezone = false; int offset_to_utc_in_minutes = 0; // Splits the string into "date" part and "time" part. { std::vector<base::StringPiece> parts; if (Tokenize(raw_value, "T", &parts) != 2) return false; date = parts[0]; time_and_tz = parts[1]; } // Parses timezone suffix on the time part if available. { std::vector<base::StringPiece> parts; if (time_and_tz[time_and_tz.size() - 1] == 'Z') { // Timezone is 'Z' (UTC) has_timezone = true; offset_to_utc_in_minutes = 0; time = time_and_tz; time.remove_suffix(1); } else if (Tokenize(time_and_tz, "+", &parts) == 2) { // Timezone is "+hh:mm" format if (!ParseTimezone(parts[1], true, &offset_to_utc_in_minutes)) return false; has_timezone = true; time = parts[0]; } else if (Tokenize(time_and_tz, "-", &parts) == 2) { // Timezone is "-hh:mm" format if (!ParseTimezone(parts[1], false, &offset_to_utc_in_minutes)) return false; has_timezone = true; time = parts[0]; } else { // No timezone (uses local timezone) time = time_and_tz; } } // Parses the date part. { std::vector<base::StringPiece> parts; if (Tokenize(date, "-", &parts) != 3) return false; if (!base::StringToInt(parts[0], &exploded.year) || !base::StringToInt(parts[1], &exploded.month) || !base::StringToInt(parts[2], &exploded.day_of_month)) { return false; } } // Parses the time part. { std::vector<base::StringPiece> parts; int num_of_token = Tokenize(time, ":", &parts); if (num_of_token != 3) return false; if (!base::StringToInt(parts[0], &exploded.hour) || !base::StringToInt(parts[1], &exploded.minute)) { return false; } std::vector<base::StringPiece> seconds_parts; int num_of_seconds_token = Tokenize(parts[2], ".", &seconds_parts); if (num_of_seconds_token >= 3) return false; if (!base::StringToInt(seconds_parts[0], &exploded.second)) return false; // Only accept milli-seconds (3-digits). if (num_of_seconds_token > 1 && seconds_parts[1].length() == 3 && !base::StringToInt(seconds_parts[1], &exploded.millisecond)) { return false; } } exploded.day_of_week = 0; if (!exploded.HasValidValues()) return false; if (has_timezone) { *parsed_time = base::Time::FromUTCExploded(exploded); if (offset_to_utc_in_minutes != 0) *parsed_time -= base::TimeDelta::FromMinutes(offset_to_utc_in_minutes); } else { *parsed_time = base::Time::FromLocalExploded(exploded); } return true; } std::string FormatTimeAsString(const base::Time& time) { if (time.is_null()) return kNullTimeString; base::Time::Exploded exploded; time.UTCExplode(&exploded); return base::StringPrintf( "%04d-%02d-%02dT%02d:%02d:%02d.%03dZ", exploded.year, exploded.month, exploded.day_of_month, exploded.hour, exploded.minute, exploded.second, exploded.millisecond); } std::string FormatTimeAsStringLocaltime(const base::Time& time) { if (time.is_null()) return kNullTimeString; base::Time::Exploded exploded; time.LocalExplode(&exploded); return base::StringPrintf( "%04d-%02d-%02dT%02d:%02d:%02d.%03d", exploded.year, exploded.month, exploded.day_of_month, exploded.hour, exploded.minute, exploded.second, exploded.millisecond); } } // namespace util } // namespace google_apis