]> rtime.felk.cvut.cz Git - sojka/lightdm.git/blob - liblightdm-gobject/session.c
Revert r2392 - it seems to have broken ABI in liblightdm-gobject
[sojka/lightdm.git] / liblightdm-gobject / session.c
1 /*
2  * Copyright (C) 2010 Robert Ancell.
3  * Author: Robert Ancell <robert.ancell@canonical.com>
4  *
5  * This library is free software; you can redistribute it and/or modify it under
6  * the terms of the GNU Lesser General Public License as published by the Free
7  * Software Foundation; either version 2 or version 3 of the License.
8  * See http://www.gnu.org/copyleft/lgpl.html the full text of the license.
9  */
10
11 #include <string.h>
12 #include <gio/gdesktopappinfo.h>
13
14 #include "configuration.h"
15 #include "lightdm/session.h"
16
17 enum {
18     PROP_KEY = 1,
19     PROP_NAME,
20     PROP_COMMENT
21 };
22
23 typedef struct
24 {
25     gchar *key;
26     gchar *type;
27     gchar *name;
28     gchar *comment;
29 } LightDMSessionPrivate;
30
31 G_DEFINE_TYPE (LightDMSession, lightdm_session, G_TYPE_OBJECT);
32
33 #define GET_PRIVATE(obj) G_TYPE_INSTANCE_GET_PRIVATE ((obj), LIGHTDM_TYPE_SESSION, LightDMSessionPrivate)
34
35 static gboolean have_sessions = FALSE;
36 static GList *local_sessions = NULL;
37 static GList *remote_sessions = NULL;
38
39 static gint
40 compare_session (gconstpointer a, gconstpointer b)
41 {
42     LightDMSessionPrivate *priv_a = GET_PRIVATE (a);
43     LightDMSessionPrivate *priv_b = GET_PRIVATE (b);
44     return strcmp (priv_a->name, priv_b->name);
45 }
46
47 static LightDMSession *
48 load_session (GKeyFile *key_file, const gchar *key, const gchar *default_type)
49 {
50     gchar *domain, *name, *type;
51     LightDMSession *session;
52     LightDMSessionPrivate *priv;
53     gchar *try_exec;
54
55     if (g_key_file_get_boolean (key_file, G_KEY_FILE_DESKTOP_GROUP, G_KEY_FILE_DESKTOP_KEY_NO_DISPLAY, NULL) ||
56         g_key_file_get_boolean (key_file, G_KEY_FILE_DESKTOP_GROUP, G_KEY_FILE_DESKTOP_KEY_HIDDEN, NULL))
57         return NULL;
58
59 #ifdef G_KEY_FILE_DESKTOP_KEY_GETTEXT_DOMAIN
60     domain = g_key_file_get_string (key_file, G_KEY_FILE_DESKTOP_GROUP, G_KEY_FILE_DESKTOP_KEY_GETTEXT_DOMAIN, NULL);
61 #else
62     domain = g_key_file_get_string (key_file, G_KEY_FILE_DESKTOP_GROUP, "X-GNOME-Gettext-Domain", NULL);
63 #endif
64     name = g_key_file_get_locale_string (key_file, G_KEY_FILE_DESKTOP_GROUP, G_KEY_FILE_DESKTOP_KEY_NAME, domain, NULL);
65     if (!name)
66     {
67         g_warning ("Ignoring session without name");
68         g_free (domain);
69         return NULL;
70     }
71
72     try_exec = g_key_file_get_locale_string (key_file, G_KEY_FILE_DESKTOP_GROUP, G_KEY_FILE_DESKTOP_KEY_TRY_EXEC, domain, NULL);
73     if (try_exec)
74     {
75         gchar *full_path;
76
77         full_path = g_find_program_in_path (try_exec);
78         g_free (try_exec);
79
80         if (!full_path)
81         {
82             g_free (name);
83             g_free (domain);
84             return NULL;
85         }
86         g_free (full_path);
87     }
88
89     type = g_key_file_get_string (key_file, G_KEY_FILE_DESKTOP_GROUP, "X-LightDM-Session-Type", NULL);
90     if (!type)
91         type = strdup (default_type);
92
93     session = g_object_new (LIGHTDM_TYPE_SESSION, NULL);
94     priv = GET_PRIVATE (session);
95
96     g_free (priv->key);
97     priv->key = g_strdup (key);
98
99     g_free (priv->type);
100     priv->type = type;
101
102     g_free (priv->name);
103     priv->name = name;
104
105     g_free (priv->comment);
106     priv->comment = g_key_file_get_locale_string (key_file, G_KEY_FILE_DESKTOP_GROUP, G_KEY_FILE_DESKTOP_KEY_COMMENT, domain, NULL);
107     if (!priv->comment)
108         priv->comment = g_strdup ("");
109
110     g_free (domain);
111
112     return session;
113 }
114
115 static GList *
116 load_sessions_dir (GList *sessions, const gchar *sessions_dir, const gchar *default_type)
117 {
118     GDir *directory;
119     GError *error = NULL;
120
121     directory = g_dir_open (sessions_dir, 0, &error);
122     if (error && !g_error_matches (error, G_FILE_ERROR, G_FILE_ERROR_NOENT))
123         g_warning ("Failed to open sessions directory: %s", error->message);
124     g_clear_error (&error);
125     if (!directory)
126         return sessions;
127
128     while (TRUE)
129     {
130         const gchar *filename;
131         gchar *path;
132         GKeyFile *key_file;
133         gboolean result;
134
135         filename = g_dir_read_name (directory);
136         if (filename == NULL)
137             break;
138
139         if (!g_str_has_suffix (filename, ".desktop"))
140             continue;
141
142         path = g_build_filename (sessions_dir, filename, NULL);
143
144         key_file = g_key_file_new ();
145         result = g_key_file_load_from_file (key_file, path, G_KEY_FILE_NONE, &error);
146         if (error)
147             g_warning ("Failed to load session file %s: %s:", path, error->message);
148         g_clear_error (&error);
149
150         if (result)
151         {
152             gchar *key;
153             LightDMSession *session;
154
155             key = g_strndup (filename, strlen (filename) - strlen (".desktop"));
156             session = load_session (key_file, key, default_type);
157             if (session)
158             {
159                 g_debug ("Loaded session %s (%s, %s)", path, GET_PRIVATE (session)->name, GET_PRIVATE (session)->comment);
160                 sessions = g_list_insert_sorted (sessions, session, compare_session);
161             }
162             else
163                 g_debug ("Ignoring session %s", path);
164             g_free (key);
165         }
166
167         g_free (path);
168         g_key_file_free (key_file);
169     }
170
171     g_dir_close (directory);
172
173     return sessions;
174 }
175
176 static GList *
177 load_sessions (const gchar *sessions_dir)
178 {
179     GList *sessions = NULL;
180     gchar **dirs;
181     int i;
182
183     dirs = g_strsplit (sessions_dir, ":", -1);
184     for (i = 0; dirs[i]; i++) 
185     {
186         const gchar *default_type = "x";
187
188         if (strcmp (dirs[i], WAYLAND_SESSIONS_DIR) == 0)
189             default_type = "wayland";
190
191         sessions = load_sessions_dir (sessions, dirs[i], default_type);
192     }
193  
194     g_strfreev (dirs);
195
196     return sessions;
197 }
198
199 static void
200 update_sessions (void)
201 {
202     gchar *sessions_dir;
203     gchar *remote_sessions_dir;
204     gchar *value;
205
206     if (have_sessions)
207         return;
208
209     sessions_dir = g_strdup (SESSIONS_DIR);
210     remote_sessions_dir = g_strdup (REMOTE_SESSIONS_DIR);
211
212     /* Use session directory from configuration */
213     config_load_from_standard_locations (config_get_instance (), NULL, NULL);
214
215     value = config_get_string (config_get_instance (), "LightDM", "sessions-directory");
216     if (value)
217     {
218         g_free (sessions_dir);
219         sessions_dir = value;
220     }
221
222     value = config_get_string (config_get_instance (), "LightDM", "remote-sessions-directory");
223     if (value)
224     {
225         g_free (remote_sessions_dir);
226         remote_sessions_dir = value;
227     }
228
229     local_sessions = load_sessions (sessions_dir);
230     remote_sessions = load_sessions (remote_sessions_dir);
231
232     g_free (sessions_dir);
233     g_free (remote_sessions_dir);
234
235     have_sessions = TRUE;
236 }
237
238 /**
239  * lightdm_get_sessions:
240  *
241  * Get the available sessions.
242  *
243  * Return value: (element-type LightDMSession) (transfer none): A list of #LightDMSession
244  **/
245 GList *
246 lightdm_get_sessions (void)
247 {
248     update_sessions ();
249     return local_sessions;
250 }
251
252 /**
253  * lightdm_get_remote_sessions:
254  *
255  * Get the available remote sessions.
256  *
257  * Return value: (element-type LightDMSession) (transfer none): A list of #LightDMSession
258  **/
259 GList *
260 lightdm_get_remote_sessions (void)
261 {
262     update_sessions ();
263     return remote_sessions;
264 }
265
266 /**
267  * lightdm_session_get_key:
268  * @session: A #LightDMSession
269  *
270  * Get the key for a session
271  *
272  * Return value: The session key
273  **/
274 const gchar *
275 lightdm_session_get_key (LightDMSession *session)
276 {
277     g_return_val_if_fail (LIGHTDM_IS_SESSION (session), NULL);
278     return GET_PRIVATE (session)->key;
279 }
280
281 /**
282  * lightdm_session_get_session_type:
283  * @session: A #LightDMSession
284  *
285  * Get the type a session
286  *
287  * Return value: The session type, e.g. x or mir
288  **/
289 const gchar *
290 lightdm_session_get_session_type (LightDMSession *session)
291 {
292     g_return_val_if_fail (LIGHTDM_IS_SESSION (session), NULL);
293     return GET_PRIVATE (session)->type;
294 }
295
296 /**
297  * lightdm_session_get_name:
298  * @session: A #LightDMSession
299  *
300  * Get the name for a session
301  *
302  * Return value: The session name
303  **/
304 const gchar *
305 lightdm_session_get_name (LightDMSession *session)
306 {
307     g_return_val_if_fail (LIGHTDM_IS_SESSION (session), NULL);
308     return GET_PRIVATE (session)->name;
309 }
310
311 /**
312  * lightdm_session_get_comment:
313  * @session: A #LightDMSession
314  *
315  * Get the comment for a session
316  *
317  * Return value: The session comment
318  **/
319 const gchar *
320 lightdm_session_get_comment (LightDMSession *session)
321 {
322     g_return_val_if_fail (LIGHTDM_IS_SESSION (session), NULL);
323     return GET_PRIVATE (session)->comment;
324 }
325
326 static void
327 lightdm_session_init (LightDMSession *session)
328 {
329 }
330
331 static void
332 lightdm_session_set_property (GObject      *object,
333                               guint         prop_id,
334                               const GValue *value,
335                               GParamSpec   *pspec)
336 {
337     G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
338 }
339
340 static void
341 lightdm_session_get_property (GObject    *object,
342                               guint       prop_id,
343                               GValue     *value,
344                               GParamSpec *pspec)
345 {
346     LightDMSession *self;
347
348     self = LIGHTDM_SESSION (object);
349
350     switch (prop_id) {
351     case PROP_KEY:
352         g_value_set_string (value, lightdm_session_get_key (self));
353         break;
354     case PROP_NAME:
355         g_value_set_string (value, lightdm_session_get_name (self));
356         break;
357     case PROP_COMMENT:
358         g_value_set_string (value, lightdm_session_get_comment (self));
359         break;
360     default:
361         G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
362         break;
363     }
364 }
365
366 static void
367 lightdm_session_finalize (GObject *object)
368 {
369     LightDMSession *self = LIGHTDM_SESSION (object);
370     LightDMSessionPrivate *priv = GET_PRIVATE (self);
371
372     g_free (priv->key);
373     g_free (priv->type);
374     g_free (priv->name);
375     g_free (priv->comment);
376 }
377
378 static void
379 lightdm_session_class_init (LightDMSessionClass *klass)
380 {
381     GObjectClass *object_class = G_OBJECT_CLASS (klass);
382
383     g_type_class_add_private (klass, sizeof (LightDMSessionPrivate));
384
385     object_class->set_property = lightdm_session_set_property;
386     object_class->get_property = lightdm_session_get_property;
387     object_class->finalize = lightdm_session_finalize;
388
389     g_object_class_install_property (object_class,
390                                      PROP_KEY,
391                                      g_param_spec_string ("key",
392                                                           "key",
393                                                           "Session key",
394                                                           NULL,
395                                                           G_PARAM_READABLE));
396     g_object_class_install_property (object_class,
397                                      PROP_NAME,
398                                      g_param_spec_string ("name",
399                                                           "name",
400                                                           "Session name",
401                                                           NULL,
402                                                           G_PARAM_READABLE));
403     g_object_class_install_property (object_class,
404                                      PROP_COMMENT,
405                                      g_param_spec_string ("comment",
406                                                           "comment",
407                                                           "Session comment",
408                                                           NULL,
409                                                           G_PARAM_READABLE));
410 }