1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chromeos/settings/timezone_settings.h"
10 #include "base/files/file_path.h"
11 #include "base/files/file_util.h"
12 #include "base/location.h"
13 #include "base/logging.h"
14 #include "base/memory/scoped_ptr.h"
15 #include "base/memory/singleton.h"
16 #include "base/observer_list.h"
17 #include "base/stl_util.h"
18 #include "base/strings/string_util.h"
19 #include "base/strings/utf_string_conversions.h"
20 #include "base/sys_info.h"
21 #include "base/task_runner.h"
22 #include "base/threading/worker_pool.h"
23 #include "chromeos/settings/timezone_settings_helper.h"
27 // The filepath to the timezone file that symlinks to the actual timezone file.
28 const char kTimezoneSymlink
[] = "/var/lib/timezone/localtime";
29 const char kTimezoneSymlink2
[] = "/var/lib/timezone/localtime2";
31 // The directory that contains all the timezone files. So for timezone
32 // "US/Pacific", the actual timezone file is: "/usr/share/zoneinfo/US/Pacific"
33 const char kTimezoneFilesDir
[] = "/usr/share/zoneinfo/";
35 // Fallback time zone ID used in case of an unexpected error.
36 const char kFallbackTimeZoneId
[] = "America/Los_Angeles";
38 // TODO(jungshik): Using Enumerate method in ICU gives 600+ timezones.
39 // Even after filtering out duplicate entries with a strict identity check,
40 // we still have 400+ zones. Relaxing the criteria for the timezone
41 // identity is likely to cut down the number to < 100. Until we
42 // come up with a better list, we hard-code the following list. It came from
43 // from Android initially, but more entries have been added.
44 static const char* kTimeZones
[] = {
48 "America/Los_Angeles",
59 "America/Mexico_City",
73 "America/Argentina/Buenos_Aires",
74 "America/Argentina/San_Luis",
78 "Atlantic/South_Georgia",
79 "Atlantic/Cape_Verde",
112 "Africa/Brazzaville",
116 "Africa/Johannesburg",
117 "Europe/Kaliningrad",
148 "Asia/Yekaterinburg",
183 "Australia/Adelaide",
185 "Australia/Brisbane",
189 "Pacific/Port_Moresby",
196 "Pacific/Kiritimati",
199 std::string
GetTimezoneIDAsString() {
200 // Compare with chromiumos/src/platform/init/ui.conf which fixes certain
201 // incorrect states of the timezone symlink on startup. Thus errors occuring
202 // here should be rather contrived.
204 // Look at kTimezoneSymlink, see which timezone we are symlinked to.
206 const ssize_t len
= readlink(kTimezoneSymlink
, buf
,
209 LOG(ERROR
) << "GetTimezoneID: Cannot read timezone symlink "
211 return std::string();
214 std::string
timezone(buf
, len
);
215 // Remove kTimezoneFilesDir from the beginning.
216 if (timezone
.find(kTimezoneFilesDir
) != 0) {
217 LOG(ERROR
) << "GetTimezoneID: Timezone symlink is wrong "
219 return std::string();
222 return timezone
.substr(strlen(kTimezoneFilesDir
));
225 void SetTimezoneIDFromString(const std::string
& id
) {
226 // Change the kTimezoneSymlink symlink to the path for this timezone.
227 // We want to do this in an atomic way. So we are going to create the symlink
228 // at kTimezoneSymlink2 and then move it to kTimezoneSymlink
230 base::FilePath
timezone_symlink(kTimezoneSymlink
);
231 base::FilePath
timezone_symlink2(kTimezoneSymlink2
);
232 base::FilePath
timezone_file(kTimezoneFilesDir
+ id
);
234 // Make sure timezone_file exists.
235 if (!base::PathExists(timezone_file
)) {
236 LOG(ERROR
) << "SetTimezoneID: Cannot find timezone file "
237 << timezone_file
.value();
241 // Delete old symlink2 if it exists.
242 base::DeleteFile(timezone_symlink2
, false);
244 // Create new symlink2.
245 if (symlink(timezone_file
.value().c_str(),
246 timezone_symlink2
.value().c_str()) == -1) {
247 LOG(ERROR
) << "SetTimezoneID: Unable to create symlink "
248 << timezone_symlink2
.value() << " to " << timezone_file
.value();
252 // Move symlink2 to symlink.
253 if (!base::ReplaceFile(timezone_symlink2
, timezone_symlink
, NULL
)) {
254 LOG(ERROR
) << "SetTimezoneID: Unable to move symlink "
255 << timezone_symlink2
.value() << " to "
256 << timezone_symlink
.value();
260 // Common code of the TimezoneSettings implementations.
261 class TimezoneSettingsBaseImpl
: public chromeos::system::TimezoneSettings
{
263 ~TimezoneSettingsBaseImpl() override
;
265 // TimezoneSettings implementation:
266 const icu::TimeZone
& GetTimezone() override
;
267 base::string16
GetCurrentTimezoneID() override
;
268 void SetTimezoneFromID(const base::string16
& timezone_id
) override
;
269 void AddObserver(Observer
* observer
) override
;
270 void RemoveObserver(Observer
* observer
) override
;
271 const std::vector
<icu::TimeZone
*>& GetTimezoneList() const override
;
274 TimezoneSettingsBaseImpl();
276 // Returns |timezone| if it is an element of |timezones_|.
277 // Otherwise, returns a timezone from |timezones_|, if such exists, that has
278 // the same rule as the given |timezone|.
279 // Otherwise, returns NULL.
280 // Note multiple timezones with the same time zone rules may exist
282 // US/Pacific == America/Los_Angeles
283 const icu::TimeZone
* GetKnownTimezoneOrNull(
284 const icu::TimeZone
& timezone
) const;
286 base::ObserverList
<Observer
> observers_
;
287 std::vector
<icu::TimeZone
*> timezones_
;
288 scoped_ptr
<icu::TimeZone
> timezone_
;
291 DISALLOW_COPY_AND_ASSIGN(TimezoneSettingsBaseImpl
);
294 // The TimezoneSettings implementation used in production.
295 class TimezoneSettingsImpl
: public TimezoneSettingsBaseImpl
{
297 // TimezoneSettings implementation:
298 void SetTimezone(const icu::TimeZone
& timezone
) override
;
300 static TimezoneSettingsImpl
* GetInstance();
303 friend struct base::DefaultSingletonTraits
<TimezoneSettingsImpl
>;
305 TimezoneSettingsImpl();
307 DISALLOW_COPY_AND_ASSIGN(TimezoneSettingsImpl
);
310 // The stub TimezoneSettings implementation used on Linux desktop.
311 class TimezoneSettingsStubImpl
: public TimezoneSettingsBaseImpl
{
313 // TimezoneSettings implementation:
314 void SetTimezone(const icu::TimeZone
& timezone
) override
;
316 static TimezoneSettingsStubImpl
* GetInstance();
319 friend struct base::DefaultSingletonTraits
<TimezoneSettingsStubImpl
>;
321 TimezoneSettingsStubImpl();
323 DISALLOW_COPY_AND_ASSIGN(TimezoneSettingsStubImpl
);
326 TimezoneSettingsBaseImpl::~TimezoneSettingsBaseImpl() {
327 STLDeleteElements(&timezones_
);
330 const icu::TimeZone
& TimezoneSettingsBaseImpl::GetTimezone() {
331 return *timezone_
.get();
334 base::string16
TimezoneSettingsBaseImpl::GetCurrentTimezoneID() {
335 return chromeos::system::TimezoneSettings::GetTimezoneID(GetTimezone());
338 void TimezoneSettingsBaseImpl::SetTimezoneFromID(
339 const base::string16
& timezone_id
) {
340 scoped_ptr
<icu::TimeZone
> timezone(icu::TimeZone::createTimeZone(
341 icu::UnicodeString(timezone_id
.c_str(), timezone_id
.size())));
342 SetTimezone(*timezone
);
345 void TimezoneSettingsBaseImpl::AddObserver(Observer
* observer
) {
346 observers_
.AddObserver(observer
);
349 void TimezoneSettingsBaseImpl::RemoveObserver(Observer
* observer
) {
350 observers_
.RemoveObserver(observer
);
353 const std::vector
<icu::TimeZone
*>&
354 TimezoneSettingsBaseImpl::GetTimezoneList() const {
358 TimezoneSettingsBaseImpl::TimezoneSettingsBaseImpl() {
359 for (size_t i
= 0; i
< arraysize(kTimeZones
); ++i
) {
360 timezones_
.push_back(icu::TimeZone::createTimeZone(
361 icu::UnicodeString(kTimeZones
[i
], -1, US_INV
)));
365 const icu::TimeZone
* TimezoneSettingsBaseImpl::GetKnownTimezoneOrNull(
366 const icu::TimeZone
& timezone
) const {
367 return chromeos::system::GetKnownTimezoneOrNull(timezone
, timezones_
);
370 void TimezoneSettingsImpl::SetTimezone(const icu::TimeZone
& timezone
) {
371 // Replace |timezone| by a known timezone with the same rules. If none exists
372 // go on with |timezone|.
373 const icu::TimeZone
* known_timezone
= GetKnownTimezoneOrNull(timezone
);
375 known_timezone
= &timezone
;
377 timezone_
.reset(known_timezone
->clone());
378 std::string id
= base::UTF16ToUTF8(GetTimezoneID(*known_timezone
));
379 VLOG(1) << "Setting timezone to " << id
;
380 // It's safe to change the timezone config files in the background as the
381 // following operations don't depend on the completion of the config change.
382 base::WorkerPool::GetTaskRunner(true /* task is slow */)->
383 PostTask(FROM_HERE
, base::Bind(&SetTimezoneIDFromString
, id
));
384 icu::TimeZone::setDefault(*known_timezone
);
385 FOR_EACH_OBSERVER(Observer
, observers_
, TimezoneChanged(*known_timezone
));
389 TimezoneSettingsImpl
* TimezoneSettingsImpl::GetInstance() {
390 return base::Singleton
<
391 TimezoneSettingsImpl
,
392 base::DefaultSingletonTraits
<TimezoneSettingsImpl
>>::get();
395 TimezoneSettingsImpl::TimezoneSettingsImpl() {
396 std::string id
= GetTimezoneIDAsString();
398 id
= kFallbackTimeZoneId
;
399 LOG(ERROR
) << "Got an empty string for timezone, default to '" << id
;
402 timezone_
.reset(icu::TimeZone::createTimeZone(
403 icu::UnicodeString::fromUTF8(id
)));
405 // Store a known timezone equivalent to id in |timezone_|.
406 const icu::TimeZone
* known_timezone
= GetKnownTimezoneOrNull(*timezone_
);
407 if (known_timezone
!= NULL
&& *known_timezone
!= *timezone_
)
408 // Not necessary to update the filesystem because |known_timezone| has the
410 timezone_
.reset(known_timezone
->clone());
412 icu::TimeZone::setDefault(*timezone_
);
413 VLOG(1) << "Timezone initially set to " << id
;
414 icu::UnicodeString resolvedId
;
415 std::string resolvedIdStr
;
416 timezone_
->getID(resolvedId
);
417 VLOG(1) << "Timezone initially resolved to "
418 << resolvedId
.toUTF8String(resolvedIdStr
);
421 void TimezoneSettingsStubImpl::SetTimezone(const icu::TimeZone
& timezone
) {
422 // Replace |timezone| by a known timezone with the same rules. If none exists
423 // go on with |timezone|.
424 const icu::TimeZone
* known_timezone
= GetKnownTimezoneOrNull(timezone
);
426 known_timezone
= &timezone
;
428 std::string id
= base::UTF16ToUTF8(GetTimezoneID(*known_timezone
));
429 VLOG(1) << "Setting timezone to " << id
;
430 timezone_
.reset(known_timezone
->clone());
431 icu::TimeZone::setDefault(*known_timezone
);
432 FOR_EACH_OBSERVER(Observer
, observers_
, TimezoneChanged(*known_timezone
));
436 TimezoneSettingsStubImpl
* TimezoneSettingsStubImpl::GetInstance() {
437 return base::Singleton
<
438 TimezoneSettingsStubImpl
,
439 base::DefaultSingletonTraits
<TimezoneSettingsStubImpl
>>::get();
442 TimezoneSettingsStubImpl::TimezoneSettingsStubImpl() {
443 timezone_
.reset(icu::TimeZone::createDefault());
444 const icu::TimeZone
* known_timezone
= GetKnownTimezoneOrNull(*timezone_
);
445 if (known_timezone
!= NULL
&& *known_timezone
!= *timezone_
)
446 timezone_
.reset(known_timezone
->clone());
454 TimezoneSettings::Observer::~Observer() {}
457 TimezoneSettings
* TimezoneSettings::GetInstance() {
458 if (base::SysInfo::IsRunningOnChromeOS()) {
459 return TimezoneSettingsImpl::GetInstance();
461 return TimezoneSettingsStubImpl::GetInstance();
466 base::string16
TimezoneSettings::GetTimezoneID(const icu::TimeZone
& timezone
) {
467 icu::UnicodeString id
;
469 return base::string16(id
.getBuffer(), id
.length());
472 } // namespace system
473 } // namespace chromeos