Blame src/shell-app-usage.c

Packit Service ed5168
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
Packit Service ed5168
Packit Service ed5168
#include "config.h"
Packit Service ed5168
Packit Service ed5168
#include <string.h>
Packit Service ed5168
#include <stdlib.h>
Packit Service ed5168
Packit Service ed5168
#include <X11/Xlib.h>
Packit Service ed5168
#include <X11/Xatom.h>
Packit Service ed5168
#include <gdk/gdk.h>
Packit Service ed5168
#include <gdk/gdkx.h>
Packit Service ed5168
#include <glib.h>
Packit Service ed5168
#include <gio/gio.h>
Packit Service ed5168
#include <meta/display.h>
Packit Service ed5168
#include <meta/group.h>
Packit Service ed5168
#include <meta/window.h>
Packit Service ed5168
Packit Service ed5168
#include "shell-app-usage.h"
Packit Service ed5168
#include "shell-window-tracker.h"
Packit Service ed5168
#include "shell-global.h"
Packit Service ed5168
Packit Service ed5168
/* This file includes modified code from
Packit Service ed5168
 * desktop-data-engine/engine-dbus/hippo-application-monitor.c
Packit Service ed5168
 * in the functions collecting application usage data.
Packit Service ed5168
 * Written by Owen Taylor, originally licensed under LGPL 2.1.
Packit Service ed5168
 * Copyright Red Hat, Inc. 2006-2008
Packit Service ed5168
 */
Packit Service ed5168
Packit Service ed5168
/**
Packit Service ed5168
 * SECTION:shell-app-usage
Packit Service ed5168
 * @short_description: Track application usage/state data
Packit Service ed5168
 *
Packit Service ed5168
 * This class maintains some usage and state statistics for
Packit Service ed5168
 * applications by keeping track of the approximate time an application's
Packit Service ed5168
 * windows are focused, as well as the last workspace it was seen on.
Packit Service ed5168
 * This time tracking is implemented by watching for focus notifications,
Packit Service ed5168
 * and computing a time delta between them.  Also we watch the
Packit Service ed5168
 * GNOME Session "StatusChanged" signal which by default is emitted after 5
Packit Service ed5168
 * minutes to signify idle.
Packit Service ed5168
 */
Packit Service ed5168
Packit Service ed5168
#define PRIVACY_SCHEMA "org.gnome.desktop.privacy"
Packit Service ed5168
#define ENABLE_MONITORING_KEY "remember-app-usage"
Packit Service ed5168
Packit Service ed5168
#define FOCUS_TIME_MIN_SECONDS 7 /* Need 7 continuous seconds of focus */
Packit Service ed5168
Packit Service ed5168
#define USAGE_CLEAN_DAYS 7 /* If after 7 days we haven't seen an app, purge it */
Packit Service ed5168
Packit Service ed5168
/* Data is saved to file SHELL_CONFIG_DIR/DATA_FILENAME */
Packit Service ed5168
#define DATA_FILENAME "application_state"
Packit Service ed5168
Packit Service ed5168
#define IDLE_TIME_TRANSITION_SECONDS 30 /* If we transition to idle, only count
Packit Service ed5168
                                         * this many seconds of usage */
Packit Service ed5168
Packit Service ed5168
/* The ranking algorithm we use is: every time an app score reaches SCORE_MAX,
Packit Service ed5168
 * divide all scores by 2. Scores are raised by 1 unit every SAVE_APPS_TIMEOUT
Packit Service ed5168
 * seconds. This mechanism allows the list to update relatively fast when
Packit Service ed5168
 * a new app is used intensively.
Packit Service ed5168
 * To keep the list clean, and avoid being Big Brother, apps that have not been
Packit Service ed5168
 * seen for a week and whose score is below SCORE_MIN are removed.
Packit Service ed5168
 */
Packit Service ed5168
Packit Service ed5168
/* How often we save internally app data, in seconds */
Packit Service ed5168
#define SAVE_APPS_TIMEOUT_SECONDS (5 * 60)
Packit Service ed5168
Packit Service ed5168
/* With this value, an app goes from bottom to top of the
Packit Service ed5168
 * usage list in 50 hours of use */
Packit Service ed5168
#define SCORE_MAX (3600 * 50 / FOCUS_TIME_MIN_SECONDS)
Packit Service ed5168
Packit Service ed5168
/* If an app's score in lower than this and the app has not been used in a week,
Packit Service ed5168
 * remove it */
Packit Service ed5168
#define SCORE_MIN (SCORE_MAX >> 3)
Packit Service ed5168
Packit Service ed5168
/* http://www.gnome.org/~mccann/gnome-session/docs/gnome-session.html#org.gnome.SessionManager.Presence */
Packit Service ed5168
#define GNOME_SESSION_STATUS_IDLE 3
Packit Service ed5168
Packit Service ed5168
typedef struct UsageData UsageData;
Packit Service ed5168
Packit Service ed5168
struct _ShellAppUsage
Packit Service ed5168
{
Packit Service ed5168
  GObject parent;
Packit Service ed5168
Packit Service ed5168
  GFile *configfile;
Packit Service ed5168
  GDBusProxy *session_proxy;
Packit Service ed5168
  GSettings *privacy_settings;
Packit Service ed5168
  guint idle_focus_change_id;
Packit Service ed5168
  guint save_id;
Packit Service ed5168
  gboolean currently_idle;
Packit Service ed5168
  gboolean enable_monitoring;
Packit Service ed5168
Packit Service ed5168
  long watch_start_time;
Packit Service ed5168
  ShellApp *watched_app;
Packit Service ed5168
Packit Service ed5168
  /* <char *appid, UsageData *usage> */
Packit Service ed5168
  GHashTable *app_usages;
Packit Service ed5168
};
Packit Service ed5168
Packit Service ed5168
G_DEFINE_TYPE (ShellAppUsage, shell_app_usage, G_TYPE_OBJECT);
Packit Service ed5168
Packit Service ed5168
/* Represents an application record for a given context */
Packit Service ed5168
struct UsageData
Packit Service ed5168
{
Packit Service ed5168
  gdouble score; /* Based on the number of times we'e seen the app and normalized */
Packit Service ed5168
  long last_seen; /* Used to clear old apps we've only seen a few times */
Packit Service ed5168
};
Packit Service ed5168
Packit Service ed5168
static void shell_app_usage_finalize (GObject *object);
Packit Service ed5168
Packit Service ed5168
static void on_session_status_changed (GDBusProxy *proxy, guint status, ShellAppUsage *self);
Packit Service ed5168
static void on_focus_app_changed (ShellWindowTracker *tracker, GParamSpec *spec, ShellAppUsage *self);
Packit Service ed5168
static void ensure_queued_save (ShellAppUsage *self);
Packit Service ed5168
Packit Service ed5168
static gboolean idle_save_application_usage (gpointer data);
Packit Service ed5168
Packit Service ed5168
static void restore_from_file (ShellAppUsage *self);
Packit Service ed5168
Packit Service ed5168
static void update_enable_monitoring (ShellAppUsage *self);
Packit Service ed5168
Packit Service ed5168
static void on_enable_monitoring_key_changed (GSettings     *settings,
Packit Service ed5168
                                              const gchar   *key,
Packit Service ed5168
                                              ShellAppUsage *self);
Packit Service ed5168
Packit Service ed5168
static long
Packit Service ed5168
get_time (void)
Packit Service ed5168
{
Packit Service ed5168
  GTimeVal tv;
Packit Service ed5168
  g_get_current_time (&tv;;
Packit Service ed5168
  return tv.tv_sec;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
shell_app_usage_class_init (ShellAppUsageClass *klass)
Packit Service ed5168
{
Packit Service ed5168
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit Service ed5168
Packit Service ed5168
  gobject_class->finalize = shell_app_usage_finalize;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static UsageData *
Packit Service ed5168
get_usage_for_app (ShellAppUsage *self,
Packit Service ed5168
                   ShellApp      *app)
Packit Service ed5168
{
Packit Service ed5168
  UsageData *usage;
Packit Service ed5168
  const char *appid = shell_app_get_id (app);
Packit Service ed5168
Packit Service ed5168
  usage = g_hash_table_lookup (self->app_usages, appid);
Packit Service ed5168
  if (usage)
Packit Service ed5168
    return usage;
Packit Service ed5168
Packit Service ed5168
  usage = g_new0 (UsageData, 1);
Packit Service ed5168
  g_hash_table_insert (self->app_usages, g_strdup (appid), usage);
Packit Service ed5168
Packit Service ed5168
  return usage;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
/* Limit the score to a certain level so that most used apps can change */
Packit Service ed5168
static void
Packit Service ed5168
normalize_usage (ShellAppUsage *self)
Packit Service ed5168
{
Packit Service ed5168
  GHashTableIter iter;
Packit Service ed5168
  UsageData *usage;
Packit Service ed5168
Packit Service ed5168
  g_hash_table_iter_init (&iter, self->app_usages);
Packit Service ed5168
Packit Service ed5168
  while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &usage))
Packit Service ed5168
    usage->score /= 2;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
increment_usage_for_app_at_time (ShellAppUsage *self,
Packit Service ed5168
                                 ShellApp      *app,
Packit Service ed5168
                                 long           time)
Packit Service ed5168
{
Packit Service ed5168
  UsageData *usage;
Packit Service ed5168
  guint elapsed;
Packit Service ed5168
  guint usage_count;
Packit Service ed5168
Packit Service ed5168
  usage = get_usage_for_app (self, app);
Packit Service ed5168
Packit Service ed5168
  usage->last_seen = time;
Packit Service ed5168
Packit Service ed5168
  elapsed = time - self->watch_start_time;
Packit Service ed5168
  usage_count = elapsed / FOCUS_TIME_MIN_SECONDS;
Packit Service ed5168
  if (usage_count > 0)
Packit Service ed5168
    {
Packit Service ed5168
      usage->score += usage_count;
Packit Service ed5168
      if (usage->score > SCORE_MAX)
Packit Service ed5168
        normalize_usage (self);
Packit Service ed5168
      ensure_queued_save (self);
Packit Service ed5168
    }
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
increment_usage_for_app (ShellAppUsage *self,
Packit Service ed5168
                         ShellApp      *app)
Packit Service ed5168
{
Packit Service ed5168
  long curtime = get_time ();
Packit Service ed5168
  increment_usage_for_app_at_time (self, app, curtime);
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
on_app_state_changed (ShellAppSystem *app_system,
Packit Service ed5168
                      ShellApp       *app,
Packit Service ed5168
                      gpointer        user_data)
Packit Service ed5168
{
Packit Service ed5168
  ShellAppUsage *self = SHELL_APP_USAGE (user_data);
Packit Service ed5168
  UsageData *usage;
Packit Service ed5168
  gboolean running;
Packit Service ed5168
Packit Service ed5168
  if (shell_app_is_window_backed (app))
Packit Service ed5168
    return;
Packit Service ed5168
Packit Service ed5168
  usage = get_usage_for_app (self, app);
Packit Service ed5168
Packit Service ed5168
  running = shell_app_get_state (app) == SHELL_APP_STATE_RUNNING;
Packit Service ed5168
Packit Service ed5168
  if (running)
Packit Service ed5168
    usage->last_seen = get_time ();
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
on_focus_app_changed (ShellWindowTracker *tracker,
Packit Service ed5168
                      GParamSpec         *spec,
Packit Service ed5168
                      ShellAppUsage      *self)
Packit Service ed5168
{
Packit Service ed5168
  if (self->watched_app != NULL)
Packit Service ed5168
    increment_usage_for_app (self, self->watched_app);
Packit Service ed5168
Packit Service ed5168
  if (self->watched_app)
Packit Service ed5168
    g_object_unref (self->watched_app);
Packit Service ed5168
Packit Service ed5168
  g_object_get (tracker, "focus-app", &(self->watched_app), NULL);
Packit Service ed5168
  self->watch_start_time = get_time ();
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
on_session_status_changed (GDBusProxy      *proxy,
Packit Service ed5168
                           guint            status,
Packit Service ed5168
                           ShellAppUsage *self)
Packit Service ed5168
{
Packit Service ed5168
  gboolean idle;
Packit Service ed5168
Packit Service ed5168
  idle = (status >= GNOME_SESSION_STATUS_IDLE);
Packit Service ed5168
  if (self->currently_idle == idle)
Packit Service ed5168
    return;
Packit Service ed5168
Packit Service ed5168
  self->currently_idle = idle;
Packit Service ed5168
  if (idle)
Packit Service ed5168
    {
Packit Service ed5168
      long end_time;
Packit Service ed5168
Packit Service ed5168
      /* The GNOME Session signal we watch is 5 minutes, but that's a long
Packit Service ed5168
       * time for this purpose.  Instead, just add a base 30 seconds.
Packit Service ed5168
       */
Packit Service ed5168
      if (self->watched_app)
Packit Service ed5168
        {
Packit Service ed5168
          end_time = self->watch_start_time + IDLE_TIME_TRANSITION_SECONDS;
Packit Service ed5168
          increment_usage_for_app_at_time (self, self->watched_app, end_time);
Packit Service ed5168
        }
Packit Service ed5168
    }
Packit Service ed5168
  else
Packit Service ed5168
    {
Packit Service ed5168
      /* Transitioning to !idle, reset the start time */
Packit Service ed5168
      self->watch_start_time = get_time ();
Packit Service ed5168
    }
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
session_proxy_signal (GDBusProxy *proxy, gchar *sender_name, gchar *signal_name, GVariant *parameters, gpointer user_data)
Packit Service ed5168
{
Packit Service ed5168
  if (g_str_equal (signal_name, "StatusChanged"))
Packit Service ed5168
    {
Packit Service ed5168
      guint status;
Packit Service ed5168
      g_variant_get (parameters, "(u)", &status);
Packit Service ed5168
      on_session_status_changed (proxy, status, SHELL_APP_USAGE (user_data));
Packit Service ed5168
    }
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
shell_app_usage_init (ShellAppUsage *self)
Packit Service ed5168
{
Packit Service ed5168
  ShellGlobal *global;
Packit Service ed5168
  char *shell_userdata_dir, *path;
Packit Service ed5168
  GDBusConnection *session_bus;
Packit Service ed5168
  ShellWindowTracker *tracker;
Packit Service ed5168
  ShellAppSystem *app_system;
Packit Service ed5168
Packit Service ed5168
  global = shell_global_get ();
Packit Service ed5168
Packit Service ed5168
  self->app_usages = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
Packit Service ed5168
Packit Service ed5168
  tracker = shell_window_tracker_get_default ();
Packit Service ed5168
  g_signal_connect (tracker, "notify::focus-app", G_CALLBACK (on_focus_app_changed), self);
Packit Service ed5168
Packit Service ed5168
  app_system = shell_app_system_get_default ();
Packit Service ed5168
  g_signal_connect (app_system, "app-state-changed", G_CALLBACK (on_app_state_changed), self);
Packit Service ed5168
Packit Service ed5168
  session_bus = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, NULL);
Packit Service ed5168
  self->session_proxy = g_dbus_proxy_new_sync (session_bus,
Packit Service ed5168
                                               G_DBUS_PROXY_FLAGS_NONE,
Packit Service ed5168
                                               NULL, /* interface info */
Packit Service ed5168
                                               "org.gnome.SessionManager",
Packit Service ed5168
                                               "/org/gnome/SessionManager/Presence",
Packit Service ed5168
                                               "org.gnome.SessionManager",
Packit Service ed5168
                                               NULL, /* cancellable */
Packit Service ed5168
                                               NULL /* error */);
Packit Service ed5168
  g_signal_connect (self->session_proxy, "g-signal", G_CALLBACK (session_proxy_signal), self);
Packit Service ed5168
  g_object_unref (session_bus);
Packit Service ed5168
Packit Service ed5168
  self->currently_idle = FALSE;
Packit Service ed5168
  self->enable_monitoring = FALSE;
Packit Service ed5168
Packit Service ed5168
  g_object_get (global, "userdatadir", &shell_userdata_dir, NULL),
Packit Service ed5168
  path = g_build_filename (shell_userdata_dir, DATA_FILENAME, NULL);
Packit Service ed5168
  g_free (shell_userdata_dir);
Packit Service ed5168
  self->configfile = g_file_new_for_path (path);
Packit Service ed5168
  g_free (path);
Packit Service ed5168
  restore_from_file (self);
Packit Service ed5168
Packit Service ed5168
  self->privacy_settings = g_settings_new(PRIVACY_SCHEMA);
Packit Service ed5168
  g_signal_connect (self->privacy_settings,
Packit Service ed5168
                    "changed::" ENABLE_MONITORING_KEY,
Packit Service ed5168
                    G_CALLBACK (on_enable_monitoring_key_changed),
Packit Service ed5168
                    self);
Packit Service ed5168
  update_enable_monitoring (self);
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
shell_app_usage_finalize (GObject *object)
Packit Service ed5168
{
Packit Service ed5168
  ShellAppUsage *self = SHELL_APP_USAGE (object);
Packit Service ed5168
Packit Service ed5168
  if (self->save_id > 0)
Packit Service ed5168
    g_source_remove (self->save_id);
Packit Service ed5168
Packit Service ed5168
  g_object_unref (self->privacy_settings);
Packit Service ed5168
Packit Service ed5168
  g_object_unref (self->configfile);
Packit Service ed5168
Packit Service ed5168
  g_object_unref (self->session_proxy);
Packit Service ed5168
Packit Service ed5168
  G_OBJECT_CLASS (shell_app_usage_parent_class)->finalize(object);
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static int
Packit Service ed5168
sort_apps_by_usage (gconstpointer a,
Packit Service ed5168
                    gconstpointer b,
Packit Service ed5168
                    gpointer      datap)
Packit Service ed5168
{
Packit Service ed5168
  ShellAppUsage *self = datap;
Packit Service ed5168
  ShellApp *app_a, *app_b;
Packit Service ed5168
  UsageData *usage_a, *usage_b;
Packit Service ed5168
Packit Service ed5168
  app_a = (ShellApp*)a;
Packit Service ed5168
  app_b = (ShellApp*)b;
Packit Service ed5168
Packit Service ed5168
  usage_a = g_hash_table_lookup (self->app_usages, shell_app_get_id (app_a));
Packit Service ed5168
  usage_b = g_hash_table_lookup (self->app_usages, shell_app_get_id (app_b));
Packit Service ed5168
Packit Service ed5168
  return usage_b->score - usage_a->score;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
/**
Packit Service ed5168
 * shell_app_usage_get_most_used:
Packit Service ed5168
 * @usage: the usage instance to request
Packit Service ed5168
 *
Packit Service ed5168
 * Returns: (element-type ShellApp) (transfer full): List of applications
Packit Service ed5168
 */
Packit Service ed5168
GSList *
Packit Service ed5168
shell_app_usage_get_most_used (ShellAppUsage   *self)
Packit Service ed5168
{
Packit Service ed5168
  GSList *apps;
Packit Service ed5168
  char *appid;
Packit Service ed5168
  ShellAppSystem *appsys;
Packit Service ed5168
  GHashTableIter iter;
Packit Service ed5168
Packit Service ed5168
  appsys = shell_app_system_get_default ();
Packit Service ed5168
Packit Service ed5168
  g_hash_table_iter_init (&iter, self->app_usages);
Packit Service ed5168
  apps = NULL;
Packit Service ed5168
  while (g_hash_table_iter_next (&iter, (gpointer *) &appid, NULL))
Packit Service ed5168
    {
Packit Service ed5168
      ShellApp *app;
Packit Service ed5168
Packit Service ed5168
      app = shell_app_system_lookup_app (appsys, appid);
Packit Service ed5168
      if (!app)
Packit Service ed5168
        continue;
Packit Service ed5168
Packit Service ed5168
      apps = g_slist_prepend (apps, g_object_ref (app));
Packit Service ed5168
    }
Packit Service ed5168
Packit Service ed5168
  apps = g_slist_sort_with_data (apps, sort_apps_by_usage, self);
Packit Service ed5168
Packit Service ed5168
  return apps;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
Packit Service ed5168
/**
Packit Service ed5168
 * shell_app_usage_compare:
Packit Service ed5168
 * @self: the usage instance to request
Packit Service ed5168
 * @id_a: ID of first app
Packit Service ed5168
 * @id_b: ID of second app
Packit Service ed5168
 *
Packit Service ed5168
 * Compare @id_a and @id_b based on frequency of use.
Packit Service ed5168
 *
Packit Service ed5168
 * Returns: -1 if @id_a ranks higher than @id_b, 1 if @id_b ranks higher
Packit Service ed5168
 *          than @id_a, and 0 if both rank equally.
Packit Service ed5168
 */
Packit Service ed5168
int
Packit Service ed5168
shell_app_usage_compare (ShellAppUsage *self,
Packit Service ed5168
                         const char    *id_a,
Packit Service ed5168
                         const char    *id_b)
Packit Service ed5168
{
Packit Service ed5168
  UsageData *usage_a, *usage_b;
Packit Service ed5168
Packit Service ed5168
  usage_a = g_hash_table_lookup (self->app_usages, id_a);
Packit Service ed5168
  usage_b = g_hash_table_lookup (self->app_usages, id_b);
Packit Service ed5168
Packit Service ed5168
  if (usage_a == NULL && usage_b == NULL)
Packit Service ed5168
    return 0;
Packit Service ed5168
  else if (usage_a == NULL)
Packit Service ed5168
    return 1;
Packit Service ed5168
  else if (usage_b == NULL)
Packit Service ed5168
    return -1;
Packit Service ed5168
Packit Service ed5168
  return usage_b->score - usage_a->score;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
ensure_queued_save (ShellAppUsage *self)
Packit Service ed5168
{
Packit Service ed5168
  if (self->save_id != 0)
Packit Service ed5168
    return;
Packit Service ed5168
  self->save_id = g_timeout_add_seconds (SAVE_APPS_TIMEOUT_SECONDS, idle_save_application_usage, self);
Packit Service ed5168
  g_source_set_name_by_id (self->save_id, "[gnome-shell] idle_save_application_usage");
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
/* Clean up apps we see rarely.
Packit Service ed5168
 * The logic behind this is that if an app was seen less than SCORE_MIN times
Packit Service ed5168
 * and not seen for a week, it can probably be forgotten about.
Packit Service ed5168
 * This should much reduce the size of the list and avoid 'pollution'. */
Packit Service ed5168
static gboolean
Packit Service ed5168
idle_clean_usage (ShellAppUsage *self)
Packit Service ed5168
{
Packit Service ed5168
  GHashTableIter iter;
Packit Service ed5168
  UsageData *usage;
Packit Service ed5168
  long current_time;
Packit Service ed5168
  long week_ago;
Packit Service ed5168
Packit Service ed5168
  current_time = get_time ();
Packit Service ed5168
  week_ago = current_time - (7 * 24 * 60 * 60);
Packit Service ed5168
Packit Service ed5168
  g_hash_table_iter_init (&iter, self->app_usages);
Packit Service ed5168
Packit Service ed5168
  while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &usage))
Packit Service ed5168
    {
Packit Service ed5168
      if ((usage->score < SCORE_MIN) &&
Packit Service ed5168
          (usage->last_seen < week_ago))
Packit Service ed5168
        g_hash_table_iter_remove (&iter);
Packit Service ed5168
    }
Packit Service ed5168
Packit Service ed5168
  return FALSE;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static gboolean
Packit Service ed5168
write_escaped (GDataOutputStream   *stream,
Packit Service ed5168
               const char          *str,
Packit Service ed5168
               GError             **error)
Packit Service ed5168
{
Packit Service ed5168
  gboolean ret;
Packit Service ed5168
  char *quoted = g_markup_escape_text (str, -1);
Packit Service ed5168
  ret = g_data_output_stream_put_string (stream, quoted, NULL, error);
Packit Service ed5168
  g_free (quoted);
Packit Service ed5168
  return ret;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static gboolean
Packit Service ed5168
write_attribute_string (GDataOutputStream *stream,
Packit Service ed5168
                        const char        *elt_name,
Packit Service ed5168
                        const char        *str,
Packit Service ed5168
                        GError           **error)
Packit Service ed5168
{
Packit Service ed5168
  gboolean ret = FALSE;
Packit Service ed5168
  char *elt;
Packit Service ed5168
Packit Service ed5168
  elt = g_strdup_printf (" %s=\"", elt_name);
Packit Service ed5168
  ret = g_data_output_stream_put_string (stream, elt, NULL, error);
Packit Service ed5168
  g_free (elt);
Packit Service ed5168
  if (!ret)
Packit Service ed5168
    goto out;
Packit Service ed5168
Packit Service ed5168
  ret = write_escaped (stream, str, error);
Packit Service ed5168
  if (!ret)
Packit Service ed5168
    goto out;
Packit Service ed5168
Packit Service ed5168
  ret = g_data_output_stream_put_string (stream, "\"", NULL, error);
Packit Service ed5168
Packit Service ed5168
out:
Packit Service ed5168
  return ret;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static gboolean
Packit Service ed5168
write_attribute_uint (GDataOutputStream *stream,
Packit Service ed5168
                      const char        *elt_name,
Packit Service ed5168
                      guint              value,
Packit Service ed5168
                      GError           **error)
Packit Service ed5168
{
Packit Service ed5168
  gboolean ret;
Packit Service ed5168
  char *buf;
Packit Service ed5168
Packit Service ed5168
  buf = g_strdup_printf ("%u", value);
Packit Service ed5168
  ret = write_attribute_string (stream, elt_name, buf, error);
Packit Service ed5168
  g_free (buf);
Packit Service ed5168
Packit Service ed5168
  return ret;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static gboolean
Packit Service ed5168
write_attribute_double (GDataOutputStream *stream,
Packit Service ed5168
                        const char        *elt_name,
Packit Service ed5168
                        double             value,
Packit Service ed5168
                        GError           **error)
Packit Service ed5168
{
Packit Service ed5168
  gchar buf[G_ASCII_DTOSTR_BUF_SIZE];
Packit Service ed5168
  gboolean ret;
Packit Service ed5168
Packit Service ed5168
  g_ascii_dtostr (buf, sizeof (buf), value);
Packit Service ed5168
  ret = write_attribute_string (stream, elt_name, buf, error);
Packit Service ed5168
Packit Service ed5168
  return ret;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
/* Save app data lists to file */
Packit Service ed5168
static gboolean
Packit Service ed5168
idle_save_application_usage (gpointer data)
Packit Service ed5168
{
Packit Service ed5168
  ShellAppUsage *self = SHELL_APP_USAGE (data);
Packit Service ed5168
  char *id;
Packit Service ed5168
  GHashTableIter iter;
Packit Service ed5168
  UsageData *usage;
Packit Service ed5168
  GFileOutputStream *output;
Packit Service ed5168
  GOutputStream *buffered_output;
Packit Service ed5168
  GDataOutputStream *data_output;
Packit Service ed5168
  GError *error = NULL;
Packit Service ed5168
Packit Service ed5168
  self->save_id = 0;
Packit Service ed5168
Packit Service ed5168
  /* Parent directory is already created by shell-global */
Packit Service ed5168
  output = g_file_replace (self->configfile, NULL, FALSE, G_FILE_CREATE_NONE, NULL, &error);
Packit Service ed5168
  if (!output)
Packit Service ed5168
    {
Packit Service ed5168
      g_debug ("Could not save applications usage data: %s", error->message);
Packit Service ed5168
      g_error_free (error);
Packit Service ed5168
      return FALSE;
Packit Service ed5168
    }
Packit Service ed5168
  buffered_output = g_buffered_output_stream_new (G_OUTPUT_STREAM (output));
Packit Service ed5168
  g_object_unref (output);
Packit Service ed5168
  data_output = g_data_output_stream_new (G_OUTPUT_STREAM (buffered_output));
Packit Service ed5168
  g_object_unref (buffered_output);
Packit Service ed5168
Packit Service ed5168
  if (!g_data_output_stream_put_string (data_output, "\n<application-state>\n", NULL, &error))
Packit Service ed5168
    goto out;
Packit Service ed5168
  if (!g_data_output_stream_put_string (data_output, "  <context id=\"\">\n", NULL, &error))
Packit Service ed5168
    goto out;
Packit Service ed5168
Packit Service ed5168
  g_hash_table_iter_init (&iter, self->app_usages);
Packit Service ed5168
Packit Service ed5168
  while (g_hash_table_iter_next (&iter, (gpointer *) &id, (gpointer *) &usage))
Packit Service ed5168
    {
Packit Service ed5168
      ShellApp *app;
Packit Service ed5168
Packit Service ed5168
      app = shell_app_system_lookup_app (shell_app_system_get_default(), id);
Packit Service ed5168
Packit Service ed5168
      if (!app)
Packit Service ed5168
        continue;
Packit Service ed5168
Packit Service ed5168
      if (!g_data_output_stream_put_string (data_output, "    
Packit Service ed5168
        goto out;
Packit Service ed5168
      if (!write_attribute_string (data_output, "id", id, &error))
Packit Service ed5168
        goto out;
Packit Service ed5168
      if (!write_attribute_double (data_output, "score", usage->score, &error))
Packit Service ed5168
        goto out;
Packit Service ed5168
      if (!write_attribute_uint (data_output, "last-seen", usage->last_seen, &error))
Packit Service ed5168
        goto out;
Packit Service ed5168
      if (!g_data_output_stream_put_string (data_output, "/>\n", NULL, &error))
Packit Service ed5168
        goto out;
Packit Service ed5168
    }
Packit Service ed5168
  if (!g_data_output_stream_put_string (data_output, "  </context>\n", NULL, &error))
Packit Service ed5168
    goto out;
Packit Service ed5168
  if (!g_data_output_stream_put_string (data_output, "</application-state>\n", NULL, &error))
Packit Service ed5168
    goto out;
Packit Service ed5168
Packit Service ed5168
out:
Packit Service ed5168
  if (!error)
Packit Service ed5168
    g_output_stream_close_async (G_OUTPUT_STREAM (data_output), 0, NULL, NULL, NULL);
Packit Service ed5168
  g_object_unref (data_output);
Packit Service ed5168
  if (error)
Packit Service ed5168
    {
Packit Service ed5168
      g_debug ("Could not save applications usage data: %s", error->message);
Packit Service ed5168
      g_error_free (error);
Packit Service ed5168
    }
Packit Service ed5168
  return FALSE;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static void
Packit Service ed5168
shell_app_usage_start_element_handler  (GMarkupParseContext *context,
Packit Service ed5168
                                        const gchar         *element_name,
Packit Service ed5168
                                        const gchar        **attribute_names,
Packit Service ed5168
                                        const gchar        **attribute_values,
Packit Service ed5168
                                        gpointer             user_data,
Packit Service ed5168
                                        GError             **error)
Packit Service ed5168
{
Packit Service ed5168
  ShellAppUsage *self = user_data;
Packit Service ed5168
Packit Service ed5168
  if (strcmp (element_name, "application-state") == 0)
Packit Service ed5168
    {
Packit Service ed5168
    }
Packit Service ed5168
  else if (strcmp (element_name, "context") == 0)
Packit Service ed5168
    {
Packit Service ed5168
    }
Packit Service ed5168
  else if (strcmp (element_name, "application") == 0)
Packit Service ed5168
    {
Packit Service ed5168
      const char **attribute;
Packit Service ed5168
      const char **value;
Packit Service ed5168
      UsageData *usage;
Packit Service ed5168
      char *appid = NULL;
Packit Service ed5168
Packit Service ed5168
      for (attribute = attribute_names, value = attribute_values; *attribute; attribute++, value++)
Packit Service ed5168
        {
Packit Service ed5168
          if (strcmp (*attribute, "id") == 0)
Packit Service ed5168
            {
Packit Service ed5168
              appid = g_strdup (*value);
Packit Service ed5168
              break;
Packit Service ed5168
            }
Packit Service ed5168
        }
Packit Service ed5168
Packit Service ed5168
      if (!appid)
Packit Service ed5168
        {
Packit Service ed5168
          g_set_error (error,
Packit Service ed5168
                       G_MARKUP_ERROR,
Packit Service ed5168
                       G_MARKUP_ERROR_PARSE,
Packit Service ed5168
                       "Missing attribute id on <%s> element",
Packit Service ed5168
                       element_name);
Packit Service ed5168
          return;
Packit Service ed5168
        }
Packit Service ed5168
Packit Service ed5168
      usage = g_new0 (UsageData, 1);
Packit Service ed5168
      g_hash_table_insert (self->app_usages, appid, usage);
Packit Service ed5168
Packit Service ed5168
      for (attribute = attribute_names, value = attribute_values; *attribute; attribute++, value++)
Packit Service ed5168
        {
Packit Service ed5168
          if (strcmp (*attribute, "score") == 0)
Packit Service ed5168
            {
Packit Service ed5168
              usage->score = g_ascii_strtod (*value, NULL);
Packit Service ed5168
            }
Packit Service ed5168
          else if (strcmp (*attribute, "last-seen") == 0)
Packit Service ed5168
            {
Packit Service ed5168
              usage->last_seen = (guint) g_ascii_strtoull (*value, NULL, 10);
Packit Service ed5168
            }
Packit Service ed5168
        }
Packit Service ed5168
    }
Packit Service ed5168
  else
Packit Service ed5168
    {
Packit Service ed5168
      g_set_error (error,
Packit Service ed5168
                   G_MARKUP_ERROR,
Packit Service ed5168
                   G_MARKUP_ERROR_PARSE,
Packit Service ed5168
                   "Unknown element <%s>",
Packit Service ed5168
                   element_name);
Packit Service ed5168
    }
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
static GMarkupParser app_state_parse_funcs =
Packit Service ed5168
{
Packit Service ed5168
  shell_app_usage_start_element_handler,
Packit Service ed5168
  NULL,
Packit Service ed5168
  NULL,
Packit Service ed5168
  NULL,
Packit Service ed5168
  NULL
Packit Service ed5168
};
Packit Service ed5168
Packit Service ed5168
/* Load data about apps usage from file */
Packit Service ed5168
static void
Packit Service ed5168
restore_from_file (ShellAppUsage *self)
Packit Service ed5168
{
Packit Service ed5168
  GFileInputStream *input;
Packit Service ed5168
  GMarkupParseContext *parse_context;
Packit Service ed5168
  GError *error = NULL;
Packit Service ed5168
  char buf[1024];
Packit Service ed5168
Packit Service ed5168
  input = g_file_read (self->configfile, NULL, &error);
Packit Service ed5168
  if (error)
Packit Service ed5168
    {
Packit Service ed5168
      if (error->code != G_IO_ERROR_NOT_FOUND)
Packit Service ed5168
        g_warning ("Could not load applications usage data: %s", error->message);
Packit Service ed5168
Packit Service ed5168
      g_error_free (error);
Packit Service ed5168
      return;
Packit Service ed5168
    }
Packit Service ed5168
Packit Service ed5168
  parse_context = g_markup_parse_context_new (&app_state_parse_funcs, 0, self, NULL);
Packit Service ed5168
Packit Service ed5168
  while (TRUE)
Packit Service ed5168
    {
Packit Service ed5168
      gssize count = g_input_stream_read ((GInputStream*) input, buf, sizeof(buf), NULL, &error);
Packit Service ed5168
      if (count <= 0)
Packit Service ed5168
        goto out;
Packit Service ed5168
      if (!g_markup_parse_context_parse (parse_context, buf, count, &error))
Packit Service ed5168
        goto out;
Packit Service ed5168
     }
Packit Service ed5168
Packit Service ed5168
out:
Packit Service ed5168
  g_markup_parse_context_free (parse_context);
Packit Service ed5168
  g_input_stream_close ((GInputStream*)input, NULL, NULL);
Packit Service ed5168
  g_object_unref (input);
Packit Service ed5168
Packit Service ed5168
  idle_clean_usage (self);
Packit Service ed5168
Packit Service ed5168
  if (error)
Packit Service ed5168
    {
Packit Service ed5168
      g_warning ("Could not load applications usage data: %s", error->message);
Packit Service ed5168
      g_error_free (error);
Packit Service ed5168
    }
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
/* Enable or disable the timers, depending on the value of ENABLE_MONITORING_KEY
Packit Service ed5168
 * and taking care of the previous state.  If selfing is disabled, we still
Packit Service ed5168
 * report apps usage based on (possibly) saved data, but don't collect data.
Packit Service ed5168
 */
Packit Service ed5168
static void
Packit Service ed5168
update_enable_monitoring (ShellAppUsage *self)
Packit Service ed5168
{
Packit Service ed5168
  gboolean enable;
Packit Service ed5168
Packit Service ed5168
  enable = g_settings_get_boolean (self->privacy_settings,
Packit Service ed5168
                                   ENABLE_MONITORING_KEY);
Packit Service ed5168
Packit Service ed5168
  /* Be sure not to start the timers if they were already set */
Packit Service ed5168
  if (enable && !self->enable_monitoring)
Packit Service ed5168
    {
Packit Service ed5168
      on_focus_app_changed (shell_window_tracker_get_default (), NULL, self);
Packit Service ed5168
    }
Packit Service ed5168
  /* ...and don't try to stop them if they were not running */
Packit Service ed5168
  else if (!enable && self->enable_monitoring)
Packit Service ed5168
    {
Packit Service ed5168
      if (self->watched_app)
Packit Service ed5168
        g_object_unref (self->watched_app);
Packit Service ed5168
      self->watched_app = NULL;
Packit Service ed5168
      if (self->save_id)
Packit Service ed5168
        {
Packit Service ed5168
          g_source_remove (self->save_id);
Packit Service ed5168
          self->save_id = 0;
Packit Service ed5168
        }
Packit Service ed5168
    }
Packit Service ed5168
Packit Service ed5168
  self->enable_monitoring = enable;
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
/* Called when the ENABLE_MONITORING_KEY boolean has changed */
Packit Service ed5168
static void
Packit Service ed5168
on_enable_monitoring_key_changed (GSettings     *settings,
Packit Service ed5168
                                  const gchar   *key,
Packit Service ed5168
                                  ShellAppUsage *self)
Packit Service ed5168
{
Packit Service ed5168
  update_enable_monitoring (self);
Packit Service ed5168
}
Packit Service ed5168
Packit Service ed5168
/**
Packit Service ed5168
 * shell_app_usage_get_default:
Packit Service ed5168
 *
Packit Service ed5168
 * Return Value: (transfer none): The global #ShellAppUsage instance
Packit Service ed5168
 */
Packit Service ed5168
ShellAppUsage *
Packit Service ed5168
shell_app_usage_get_default (void)
Packit Service ed5168
{
Packit Service ed5168
  static ShellAppUsage *instance;
Packit Service ed5168
Packit Service ed5168
  if (instance == NULL)
Packit Service ed5168
    instance = g_object_new (SHELL_TYPE_APP_USAGE, NULL);
Packit Service ed5168
Packit Service ed5168
  return instance;
Packit Service ed5168
}