472 lines
15 KiB
C
472 lines
15 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
|
|
*
|
|
* Copyright (C) 2013 Kalev Lember <kalevlember@gmail.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gsd-timezone-monitor.h"
|
|
|
|
#include "timedated.h"
|
|
#include "tz.h"
|
|
#include "weather-tz.h"
|
|
|
|
#include <geoclue.h>
|
|
#include <geocode-glib/geocode-glib.h>
|
|
#include <polkit/polkit.h>
|
|
|
|
#define DESKTOP_ID "gnome-datetime-panel"
|
|
#define SET_TIMEZONE_PERMISSION "org.freedesktop.timedate1.set-timezone"
|
|
|
|
enum {
|
|
TIMEZONE_CHANGED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static int signals[LAST_SIGNAL] = { 0 };
|
|
|
|
typedef struct
|
|
{
|
|
GCancellable *cancellable;
|
|
GPermission *permission;
|
|
Timedate1 *dtm;
|
|
|
|
GClueClient *geoclue_client;
|
|
GClueSimple *geoclue_simple;
|
|
GCancellable *geoclue_cancellable;
|
|
|
|
gchar *current_timezone;
|
|
|
|
GSettings *location_settings;
|
|
} GsdTimezoneMonitorPrivate;
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (GsdTimezoneMonitor, gsd_timezone_monitor, G_TYPE_OBJECT)
|
|
|
|
static void
|
|
set_timezone_cb (GObject *source,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
GsdTimezoneMonitorPrivate *priv;
|
|
GError *error = NULL;
|
|
|
|
if (!timedate1_call_set_timezone_finish (TIMEDATE1 (source),
|
|
res,
|
|
&error)) {
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
|
g_warning ("Could not set system timezone: %s", error->message);
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
priv = gsd_timezone_monitor_get_instance_private (user_data);
|
|
g_signal_emit (G_OBJECT (user_data),
|
|
signals[TIMEZONE_CHANGED],
|
|
0, priv->current_timezone);
|
|
|
|
g_debug ("Successfully changed timezone to '%s'",
|
|
priv->current_timezone);
|
|
}
|
|
|
|
static void
|
|
queue_set_timezone (GsdTimezoneMonitor *self,
|
|
const gchar *new_timezone)
|
|
{
|
|
GsdTimezoneMonitorPrivate *priv = gsd_timezone_monitor_get_instance_private (self);
|
|
|
|
g_debug ("Changing timezone to '%s'", new_timezone);
|
|
|
|
timedate1_call_set_timezone (priv->dtm,
|
|
new_timezone,
|
|
TRUE,
|
|
priv->cancellable,
|
|
set_timezone_cb,
|
|
self);
|
|
|
|
g_free (priv->current_timezone);
|
|
priv->current_timezone = g_strdup (new_timezone);
|
|
}
|
|
|
|
static gint
|
|
compare_locations (TzLocation *a,
|
|
TzLocation *b)
|
|
{
|
|
if (a->dist > b->dist)
|
|
return 1;
|
|
|
|
if (a->dist < b->dist)
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static GList *
|
|
sort_by_closest_to (GList *locations,
|
|
GeocodeLocation *location)
|
|
{
|
|
GList *l;
|
|
|
|
for (l = locations; l; l = l->next) {
|
|
GeocodeLocation *loc;
|
|
TzLocation *tz_location = l->data;
|
|
|
|
loc = geocode_location_new (tz_location->latitude,
|
|
tz_location->longitude,
|
|
GEOCODE_LOCATION_ACCURACY_UNKNOWN);
|
|
tz_location->dist = geocode_location_get_distance_from (loc, location);
|
|
g_object_unref (loc);
|
|
}
|
|
|
|
return g_list_sort (locations, (GCompareFunc) compare_locations);
|
|
}
|
|
|
|
static GList *
|
|
ptr_array_to_list (GPtrArray *array)
|
|
{
|
|
GList *l = NULL;
|
|
gint i;
|
|
|
|
for (i = 0; i < array->len; i++)
|
|
l = g_list_prepend (l, g_ptr_array_index (array, i));
|
|
|
|
return l;
|
|
}
|
|
|
|
static GList *
|
|
find_by_country (GList *locations,
|
|
const gchar *country_code)
|
|
{
|
|
GList *l, *found = NULL;
|
|
gchar *c1;
|
|
gchar *c2;
|
|
|
|
c1 = g_ascii_strdown (country_code, -1);
|
|
|
|
for (l = locations; l; l = l->next) {
|
|
TzLocation *loc = l->data;
|
|
|
|
c2 = g_ascii_strdown (loc->country, -1);
|
|
if (g_strcmp0 (c1, c2) == 0)
|
|
found = g_list_prepend (found, loc);
|
|
g_free (c2);
|
|
}
|
|
g_free (c1);
|
|
|
|
return found;
|
|
}
|
|
|
|
static gchar *
|
|
find_timezone (GsdTimezoneMonitor *self,
|
|
GeocodeLocation *location,
|
|
const gchar *country_code)
|
|
{
|
|
TzDB *tzdb;
|
|
gchar *res;
|
|
GList *filtered;
|
|
GList *weather_locations;
|
|
GList *locations;
|
|
TzLocation *closest_tz_location;
|
|
|
|
tzdb = tz_load_db ();
|
|
|
|
/* First load locations from Olson DB */
|
|
locations = ptr_array_to_list (tz_get_locations (tzdb));
|
|
g_return_val_if_fail (locations != NULL, NULL);
|
|
|
|
/* ... and then add libgweather's locations as well */
|
|
weather_locations = weather_tz_db_get_locations (country_code);
|
|
locations = g_list_concat (locations,
|
|
g_list_copy (weather_locations));
|
|
|
|
/* Filter tz locations by country */
|
|
filtered = find_by_country (locations, country_code);
|
|
if (filtered != NULL) {
|
|
g_list_free (locations);
|
|
locations = filtered;
|
|
} else {
|
|
g_debug ("No match for country code '%s' in tzdb", country_code);
|
|
}
|
|
|
|
/* Find the closest tz location */
|
|
locations = sort_by_closest_to (locations, location);
|
|
closest_tz_location = (TzLocation *) locations->data;
|
|
|
|
res = g_strdup (closest_tz_location->zone);
|
|
|
|
g_list_free (locations);
|
|
g_list_free_full (weather_locations, (GDestroyNotify) tz_location_free);
|
|
tz_db_free (tzdb);
|
|
|
|
return res;
|
|
}
|
|
|
|
static void
|
|
process_location (GsdTimezoneMonitor *self,
|
|
GeocodePlace *place)
|
|
{
|
|
GeocodeLocation *location;
|
|
GsdTimezoneMonitorPrivate *priv = gsd_timezone_monitor_get_instance_private (self);
|
|
const gchar *country_code;
|
|
g_autofree gchar *new_timezone = NULL;
|
|
|
|
country_code = geocode_place_get_country_code (place);
|
|
location = geocode_place_get_location (place);
|
|
|
|
new_timezone = find_timezone (self, location, country_code);
|
|
|
|
if (g_strcmp0 (priv->current_timezone, new_timezone) != 0) {
|
|
g_debug ("Found updated timezone '%s' for country '%s'",
|
|
new_timezone, country_code);
|
|
queue_set_timezone (self, new_timezone);
|
|
} else {
|
|
g_debug ("Timezone didn't change from '%s' for country '%s'",
|
|
new_timezone, country_code);
|
|
}
|
|
}
|
|
|
|
static void
|
|
on_reverse_geocoding_ready (GObject *source_object,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
GeocodePlace *place;
|
|
GError *error = NULL;
|
|
|
|
place = geocode_reverse_resolve_finish (GEOCODE_REVERSE (source_object),
|
|
res,
|
|
&error);
|
|
if (error != NULL) {
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
|
g_debug ("Reverse geocoding failed: %s", error->message);
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
g_debug ("Geocode lookup resolved country to '%s'",
|
|
geocode_place_get_country (place));
|
|
|
|
process_location (user_data, place);
|
|
g_object_unref (place);
|
|
}
|
|
|
|
static void
|
|
start_reverse_geocoding (GsdTimezoneMonitor *self,
|
|
gdouble latitude,
|
|
gdouble longitude)
|
|
{
|
|
GeocodeLocation *location;
|
|
GeocodeReverse *reverse;
|
|
GsdTimezoneMonitorPrivate *priv = gsd_timezone_monitor_get_instance_private (self);
|
|
|
|
location = geocode_location_new (latitude,
|
|
longitude,
|
|
GEOCODE_LOCATION_ACCURACY_CITY);
|
|
|
|
reverse = geocode_reverse_new_for_location (location);
|
|
geocode_reverse_resolve_async (reverse,
|
|
priv->geoclue_cancellable,
|
|
on_reverse_geocoding_ready,
|
|
self);
|
|
|
|
g_object_unref (location);
|
|
g_object_unref (reverse);
|
|
}
|
|
|
|
static void
|
|
on_location_notify (GClueSimple *simple,
|
|
GParamSpec *pspec,
|
|
gpointer user_data)
|
|
{
|
|
GsdTimezoneMonitor *self = user_data;
|
|
GClueLocation *location;
|
|
gdouble latitude, longitude;
|
|
|
|
location = gclue_simple_get_location (simple);
|
|
|
|
latitude = gclue_location_get_latitude (location);
|
|
longitude = gclue_location_get_longitude (location);
|
|
|
|
g_debug ("Got location %lf,%lf", latitude, longitude);
|
|
|
|
start_reverse_geocoding (self, latitude, longitude);
|
|
}
|
|
|
|
static void
|
|
on_geoclue_simple_ready (GObject *source_object,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
GError *error = NULL;
|
|
GsdTimezoneMonitorPrivate *priv;
|
|
GClueSimple *geoclue_simple;
|
|
|
|
geoclue_simple = gclue_simple_new_finish (res, &error);
|
|
if (geoclue_simple == NULL) {
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
|
g_warning ("Failed to connect to GeoClue2 service: %s", error->message);
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
g_debug ("Geoclue now available");
|
|
|
|
priv = gsd_timezone_monitor_get_instance_private (user_data);
|
|
priv->geoclue_simple = geoclue_simple;
|
|
priv->geoclue_client = gclue_simple_get_client (priv->geoclue_simple);
|
|
gclue_client_set_distance_threshold (priv->geoclue_client,
|
|
GEOCODE_LOCATION_ACCURACY_CITY);
|
|
|
|
g_signal_connect (priv->geoclue_simple, "notify::location",
|
|
G_CALLBACK (on_location_notify), user_data);
|
|
|
|
on_location_notify (priv->geoclue_simple, NULL, user_data);
|
|
}
|
|
|
|
static void
|
|
start_geoclue (GsdTimezoneMonitor *self)
|
|
{
|
|
GsdTimezoneMonitorPrivate *priv = gsd_timezone_monitor_get_instance_private (self);
|
|
|
|
g_debug ("Timezone monitor enabled, starting geoclue");
|
|
|
|
priv->geoclue_cancellable = g_cancellable_new ();
|
|
gclue_simple_new (DESKTOP_ID,
|
|
GCLUE_ACCURACY_LEVEL_CITY,
|
|
priv->geoclue_cancellable,
|
|
on_geoclue_simple_ready,
|
|
self);
|
|
|
|
}
|
|
|
|
static void
|
|
stop_geoclue (GsdTimezoneMonitor *self)
|
|
{
|
|
GsdTimezoneMonitorPrivate *priv = gsd_timezone_monitor_get_instance_private (self);
|
|
|
|
g_debug ("Timezone monitor disabled, stopping geoclue");
|
|
|
|
g_cancellable_cancel (priv->geoclue_cancellable);
|
|
g_clear_object (&priv->geoclue_cancellable);
|
|
|
|
if (priv->geoclue_client) {
|
|
gclue_client_call_stop (priv->geoclue_client, NULL, NULL, NULL);
|
|
priv->geoclue_client = NULL;
|
|
}
|
|
|
|
g_clear_object (&priv->geoclue_simple);
|
|
}
|
|
|
|
GsdTimezoneMonitor *
|
|
gsd_timezone_monitor_new (void)
|
|
{
|
|
return g_object_new (GSD_TYPE_TIMEZONE_MONITOR, NULL);
|
|
}
|
|
|
|
static void
|
|
gsd_timezone_monitor_finalize (GObject *obj)
|
|
{
|
|
GsdTimezoneMonitor *monitor = GSD_TIMEZONE_MONITOR (obj);
|
|
GsdTimezoneMonitorPrivate *priv = gsd_timezone_monitor_get_instance_private (monitor);
|
|
|
|
g_debug ("Stopping timezone monitor");
|
|
|
|
stop_geoclue (monitor);
|
|
|
|
if (priv->cancellable) {
|
|
g_cancellable_cancel (priv->cancellable);
|
|
g_clear_object (&priv->cancellable);
|
|
}
|
|
|
|
g_clear_object (&priv->dtm);
|
|
g_clear_object (&priv->permission);
|
|
g_clear_pointer (&priv->current_timezone, g_free);
|
|
|
|
g_clear_object (&priv->location_settings);
|
|
|
|
G_OBJECT_CLASS (gsd_timezone_monitor_parent_class)->finalize (obj);
|
|
}
|
|
|
|
static void
|
|
gsd_timezone_monitor_class_init (GsdTimezoneMonitorClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->finalize = gsd_timezone_monitor_finalize;
|
|
|
|
signals[TIMEZONE_CHANGED] =
|
|
g_signal_new ("timezone-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GsdTimezoneMonitorClass, timezone_changed),
|
|
NULL, NULL,
|
|
NULL,
|
|
G_TYPE_NONE, 1, G_TYPE_POINTER);
|
|
}
|
|
|
|
static void
|
|
check_location_settings (GsdTimezoneMonitor *self)
|
|
{
|
|
GsdTimezoneMonitorPrivate *priv = gsd_timezone_monitor_get_instance_private (self);
|
|
if (g_settings_get_boolean (priv->location_settings, "enabled"))
|
|
start_geoclue (self);
|
|
else
|
|
stop_geoclue (self);
|
|
}
|
|
|
|
static void
|
|
gsd_timezone_monitor_init (GsdTimezoneMonitor *self)
|
|
{
|
|
GError *error = NULL;
|
|
GsdTimezoneMonitorPrivate *priv = gsd_timezone_monitor_get_instance_private (self);
|
|
|
|
g_debug ("Starting timezone monitor");
|
|
|
|
priv->permission = polkit_permission_new_sync (SET_TIMEZONE_PERMISSION,
|
|
NULL, NULL,
|
|
&error);
|
|
if (priv->permission == NULL) {
|
|
g_warning ("Could not get '%s' permission: %s",
|
|
SET_TIMEZONE_PERMISSION,
|
|
error->message);
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
if (!g_permission_get_allowed (priv->permission)) {
|
|
g_debug ("No permission to set timezone");
|
|
return;
|
|
}
|
|
|
|
priv->cancellable = g_cancellable_new ();
|
|
priv->dtm = timedate1_proxy_new_for_bus_sync (G_BUS_TYPE_SYSTEM,
|
|
G_DBUS_PROXY_FLAGS_NONE,
|
|
"org.freedesktop.timedate1",
|
|
"/org/freedesktop/timedate1",
|
|
priv->cancellable,
|
|
&error);
|
|
if (priv->dtm == NULL) {
|
|
g_warning ("Could not get proxy for DateTimeMechanism: %s", error->message);
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
priv->current_timezone = timedate1_dup_timezone (priv->dtm);
|
|
|
|
priv->location_settings = g_settings_new ("org.gnome.system.location");
|
|
g_signal_connect_swapped (priv->location_settings, "changed::enabled",
|
|
G_CALLBACK (check_location_settings), self);
|
|
check_location_settings (self);
|
|
}
|