This repository has been archived by the owner on Oct 15, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 123
/
Copy pathelektrasettingsbackend.c
669 lines (602 loc) · 24.6 KB
/
elektrasettingsbackend.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
#define G_SETTINGS_ENABLE_BACKEND
#include <gio/gio.h>
#include <gio/gsettingsbackend.h>
#include <glib.h>
#include <elektra/gelektra-kdb.h>
#include <elektra/gelektra-key.h>
#include <elektra/gelektra-keyset.h>
#undef G_LOG_DOMAIN
#define G_LOG_DOMAIN "ElektraSettings"
#define G_ELEKTRA_TEST_STRING "test"
#ifndef G_ELEKTRA_SETTINGS_MODULE_PRIORITY
#error "no gsetting priority selected"
#endif
#define G_ELEKTRA_SETTINGS_SYSTEM "system:/"
#define G_ELEKTRA_SETTINGS_USER "user:/"
#ifndef G_ELEKTRA_SETTINGS_PATH
#define G_ELEKTRA_SETTINGS_PATH "sw"
#endif
typedef GSettingsBackendClass ElektraSettingsBackendClass;
typedef struct
{
GSettingsBackend backend;
/*< private >*/
GElektraKdb * gkdb;
GElektraKey * gkey_user;
GElektraKey * gkey_system;
GElektraKey * gkey_error;
GElektraKeySet * gks_user;
GElektraKeySet * gks_system;
GElektraKeySet * subscription_gks_keys;
GElektraKeySet * subscription_gks_paths;
GDBusConnection * dbus_connections[2];
} ElektraSettingsBackend;
/**
* SECTION:elektrasettingsbackend
* @title: ElektraSettingsBackend
* @short_description: Implementation of the GSettingsBackend Interface with Elektra
* @include: gio/elektrasettingsbackend.h
* @see_also: #GSettingsBackend #GSettings, #GIOExtensionPoint
*
* Description of the GSettingsBackend:
*
* The #GSettingsBackend interface defines a generic interface for
* non-strictly-typed data that is stored in a hierarchy. To implement
* an alternative storage backend for #GSettings, you need to implement
* the #GSettingsBackend interface and then make it implement the
* extension point #G_SETTINGS_BACKEND_EXTENSION_POINT_NAME.
*
* The interface defines methods for reading and writing values, a
* method for determining if writing of certain values will fail
* (lockdown) and a change notification mechanism.
*
* The semantics of the interface are very precisely defined and
* implementations must carefully adhere to the expectations of
* callers that are documented on each of the interface methods.
*
* Some of the GSettingsBackend functions accept or return a #GTree.
* These trees always have strings as keys and #GVariant as values.
* g_settings_backend_create_tree() is a convenience function to create
* suitable trees.
*
* The GSettingsBackend API is exported to allow third-party
* implementations, but does not carry the same stability guarantees
* as the public GIO API. For this reason, you have to define the
* C preprocessor symbol %G_SETTINGS_ENABLE_BACKEND before including
* `gio/gsettingsbackend.h`.
**/
static GType elektra_settings_backend_get_type (void);
G_DEFINE_TYPE (ElektraSettingsBackend, elektra_settings_backend, G_TYPE_SETTINGS_BACKEND)
/* elektra_settings_backend_sync implements g_settings_backend_sync:
* @backend: a #GSettingsBackend
*
* Write and read changes.
*/
static void elektra_settings_backend_sync (GSettingsBackend * backend)
{
// TODO: use three-way merge when ready
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
if (gelektra_kdb_set (esb->gkdb, esb->gks_user, esb->gkey_user) == -1 ||
gelektra_kdb_get (esb->gkdb, esb->gks_user, esb->gkey_user) == -1)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s\n", "Error on sync!");
return;
}
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s\n", "Sync state");
}
static GVariant * elektra_settings_read_string (GElektraKdb * kdb, GElektraKeySet * ks, GElektraKey * parentKey, gchar * keypathname,
const GVariantType * expected_type)
{
gelektra_kdb_get (kdb, ks, parentKey);
/* Lookup the requested key */
GElektraKey * gkey = gelektra_keyset_lookup_byname (ks, keypathname, GELEKTRA_KDB_O_NONE);
/* free the passed path string */
g_free (keypathname);
if (gkey == NULL)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Key with path could not be found in Elekras kdb");
return NULL;
}
else
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s ", "Key found");
GVariant * read_gvariant;
GError * err = NULL;
gchar * string_value = g_malloc (gelektra_key_getvaluesize (gkey));
if (gelektra_key_getstring (gkey, string_value, gelektra_key_getvaluesize (gkey)) == -1)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s!", "but we could not read the string from Elektra kdb");
return NULL;
}
/* now parse it with the expected type from GSettings */
read_gvariant = g_variant_parse (expected_type, string_value, NULL, NULL, &err);
if (err != NULL)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s!", "but GVariant error on parsing string value:", err->message);
g_error_free (err);
return NULL;
}
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s.", "and GVariant parsed value is:", string_value);
return read_gvariant;
}
}
static gboolean elektra_settings_write_string (GSettingsBackend * backend, gchar * keypathname, GVariant * value)
{
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
/* Lookup if key already exists */
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s: %s.", "ksLookup keypathname", keypathname);
GElektraKey * gkey = gelektra_keyset_lookup_byname (esb->gks_user, keypathname, GELEKTRA_KDB_O_NONE);
gchar * string_value = (value != NULL ? g_variant_print ((GVariant *) value, FALSE) : NULL);
if (gkey == NULL)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s %s.", "Key not found, creating new key:", keypathname, string_value);
gkey = gelektra_key_new (keypathname, KEY_VALUE, string_value, KEY_END);
g_free (keypathname);
if (gkey == NULL)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Error douring key creation");
return FALSE;
}
if (gelektra_keyset_append (esb->gks_user, gkey) == -1)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Could not append the new key!");
}
}
else
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s %s %s.", "Found key:", keypathname, "and set value to", string_value);
g_free (keypathname);
gelektra_key_setstring (gkey, string_value);
}
return TRUE;
}
/* elektra_settings_backend_read implements g_settings_backend_read:
* @backend: a #GSettingsBackend implementation
* @key: the key to read
* @expected_type: a #GVariantType
* @default_value: if the default value should be returned
*
* Reads a key. This call will never block.
*
* If the key exists, the value associated with it will be returned.
* If the key does not exist, %NULL will be returned.
*
* The returned value will be of the type given in @expected_type. If
* the backend stored a value of a different type then %NULL will be
* returned.
*
* If @default_value is %TRUE then this gets the default value from the
* backend (ie: the one that the backend would contain if
* g_settings_reset() were called).
*
* Returns: the value that was read, or %NULL
*/
static GVariant * elektra_settings_backend_read (GSettingsBackend * backend, const gchar * key, const GVariantType * expected_type,
gboolean default_value)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s %s %.*s %s %s%s", "function read:", key,
"expected_type is:", (int) (g_variant_type_get_string_length (expected_type) & INT_MAX),
g_variant_type_peek_string (expected_type), "and we", (default_value ? "" : "do not "), "want the default_value");
GVariant * ret = 0;
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
if (default_value)
{
gchar * path = g_strconcat (G_ELEKTRA_SETTINGS_SYSTEM, G_ELEKTRA_SETTINGS_PATH, key, NULL);
ret = elektra_settings_read_string (esb->gkdb, esb->gks_system, esb->gkey_system, path, expected_type);
}
else
{
gchar * path = g_strconcat (G_ELEKTRA_SETTINGS_USER, G_ELEKTRA_SETTINGS_PATH, key, NULL);
ret = elektra_settings_read_string (esb->gkdb, esb->gks_user, esb->gkey_user, path, expected_type);
}
return ret;
}
/* elektra_settings_backend_read_user_value implements g_settings_backend_read_user_value:
* @backend: a #GSettingsBackend implementation
* @key: the key to read
* @expected_type: a #GVariantType
*
* Reads the 'user value' of a key.
*
* This is the value of the key that the user has control over and has
* set for themselves. Put another way: if the user did not set the
* value for themselves, then this will return %NULL (even if the
* sysadmin has provided a default value).
*
* Returns: the value that was read, or %NULL
*/
static GVariant * elektra_settings_backend_read_user_value (GSettingsBackend * backend, const gchar * key,
const GVariantType * expected_type)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s. %s %s.", "Function read_user_value:", key,
"Expected_type is:", g_variant_type_peek_string (expected_type));
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
gchar * path = g_strconcat (G_ELEKTRA_SETTINGS_USER, G_ELEKTRA_SETTINGS_PATH, key, NULL);
GVariant * ret = elektra_settings_read_string (esb->gkdb, esb->gks_user, esb->gkey_user, path, expected_type);
return ret;
}
/* elektra_settings_backend_write implements g_settings_backend_write:
* @backend: a #GSettingsBackend implementation
* @key: the name of the key
* @value: a #GVariant value to write to this key
* @origin_tag: the origin tag
*
* Writes exactly one key.
*
* This call does not fail. During this call a
* #GSettingsBackend::changed signal will be emitted if the value of the
* key has changed. The updated key value will be visible to any signal
* callbacks.
*
* One possible method that an implementation might deal with failures is
* to emit a second "changed" signal (either during this call, or later)
* to indicate that the affected keys have suddenly "changed back" to their
* old values.
*
* Returns: %TRUE if the write succeeded, %FALSE if the key was not writable
*/
static gboolean elektra_settings_backend_write (GSettingsBackend * backend, const gchar * key, GVariant * value, gpointer origin_tag)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s %s %s", "Function write_key: ", key, "value is:", g_variant_print (value, TRUE));
gboolean ret =
elektra_settings_write_string (backend, g_strconcat (G_ELEKTRA_SETTINGS_USER, G_ELEKTRA_SETTINGS_PATH, key, NULL), value);
elektra_settings_backend_sync (backend);
// Notify GSettings that the key has changed
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s: %s", "Calling g_settings_backend_changed, Key", key);
g_settings_backend_changed (G_SETTINGS_BACKEND (backend), key, origin_tag);
return ret;
}
/* < private >
* elektra_settings_keyset_from_tree:
* @key: path of the GSettings key
* @value: GVariant value of the key
* @data: GElektraKeySet to append/write the key
*
* Writes one or more keys from a GSettings GTree to a GElektraKeySet
*
* A GSettings GTree consists of GSetting paths as keys and GVariants as values.
*
* Each key is looked up and created if needed.
*/
static gint elektra_settings_keyset_from_tree (gpointer key, gpointer value, gpointer data)
{
gchar * fullpathname = g_strconcat (G_ELEKTRA_SETTINGS_USER, G_ELEKTRA_SETTINGS_PATH, (gchar *) (key), NULL);
gchar * string_value = (value != NULL ? g_variant_print ((GVariant *) value, FALSE) : NULL);
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s: %s.", "Append to keyset ", fullpathname, string_value);
GElektraKeySet * gks = (GElektraKeySet *) data;
GElektraKey * gkey = gelektra_keyset_lookup_byname (gks, fullpathname, GELEKTRA_KDB_O_NONE);
if (gkey == NULL)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Key is new, need to create it");
gkey = gelektra_key_new (fullpathname, KEY_VALUE, string_value, KEY_END);
if (gkey == NULL)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Error douring key creation");
return FALSE;
}
if (gelektra_keyset_append (gks, gkey) == -1)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Could not append the new key!");
return FALSE;
}
}
else
{
gelektra_key_setstring (gkey, string_value);
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Key was found and new value set");
}
return FALSE;
}
/* elektra_settings_backend_write_tree implements g_settings_backend_write_tree:
* @backend: a #GSettingsBackend implementation
* @tree: a #GTree containing key-value pairs to write
* @origin_tag: the origin tag
*
* Writes one or more keys. This call will never block.
*
* The key of each item in the tree is the key name to write to and the
* value is a #GVariant to write. The proper type of #GTree for this
* call can be created with g_settings_backend_create_tree(). This call
* might take a reference to the tree; you must not modified the #GTree
* after passing it to this call.
*
* This call does not fail. During this call a #GSettingsBackend::changed
* signal will be emitted if any keys have been changed. The new values of
* all updated keys will be visible to any signal callbacks.
*
* One possible method that an implementation might deal with failures is
* to emit a second "changed" signal (either during this call, or later)
* to indicate that the affected keys have suddenly "changed back" to their
* old values.
*/
static gboolean elektra_settings_backend_write_tree (GSettingsBackend * backend, GTree * tree, gpointer origin_tag)
{
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s.", "Function writeTree. ", "We have to loop the tree and add the keys");
g_tree_foreach (tree, elektra_settings_keyset_from_tree, esb->gks_user);
elektra_settings_backend_sync (backend);
/* Notify the GSettings about the changed tree */
g_settings_backend_changed_tree (G_SETTINGS_BACKEND (backend), tree, origin_tag);
return TRUE;
}
/* elektra_settings_backend_reset implements g_settings_backend_reset:
* @backend: a #GSettingsBackend implementation
* @key: the name of a key
* @origin_tag: the origin tag
*
* "Resets" the named key to its "default" value (ie: after system-wide
* defaults, mandatory keys, etc. have been taken into account) or possibly
* unsets it.
*/
static void elektra_settings_backend_reset (GSettingsBackend * backend, const gchar * key, gpointer origin_tag)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s.", "Function reset:", key);
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
gchar * keypathname = g_strconcat (G_ELEKTRA_SETTINGS_USER, G_ELEKTRA_SETTINGS_PATH, key, NULL);
GElektraKey * gkey = gelektra_keyset_lookup_byname (esb->gks_user, keypathname, GELEKTRA_KDB_O_NONE);
g_free (keypathname);
if (gkey != NULL)
{
gelektra_keyset_lookup (esb->gks_user, gkey, GELEKTRA_KDB_O_POP);
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s: %s.", "Key found and value reset", key);
g_settings_backend_changed (G_SETTINGS_BACKEND (backend), key, origin_tag);
}
else
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Key not found, nothing to be done");
}
}
/* elektra_settings_backend_get_writable implements g_settings_backend_get_writable:
* @backend: a #GSettingsBackend implementation
* @key: the name of a key
*
* Finds out if a key is available for writing to. This is the
* interface through which 'lockdown' is implemented. Locked down
* keys will have %FALSE returned by this call.
*
* You should not write to locked-down keys, but if you do, the
* implementation will deal with it.
*
* Returns: %TRUE if the key is writable
*/
// NOTE elektra does not have a clear definition of what is writable or not
static gboolean elektra_settings_backend_get_writable (GSettingsBackend * backend, const gchar * name)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s.", "Function get_writable:", name);
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
gchar * pathToWrite = g_strconcat (G_ELEKTRA_SETTINGS_USER, G_ELEKTRA_SETTINGS_PATH, name, NULL);
GElektraKey * gkey = gelektra_keyset_lookup_byname (esb->gks_user, pathToWrite, GELEKTRA_KDB_O_NONE);
if (gkey == NULL) gkey = gelektra_key_new (pathToWrite, KEY_VALUE, G_ELEKTRA_TEST_STRING, KEY_END);
g_free (pathToWrite);
if (gkey == NULL)
{
return FALSE;
}
return TRUE;
}
static void elektra_settings_key_changed (GDBusConnection * connection G_GNUC_UNUSED, const gchar * sender_name G_GNUC_UNUSED,
const gchar * object_path G_GNUC_UNUSED, const gchar * interface_name G_GNUC_UNUSED,
const gchar * /*signal_*/ name G_GNUC_UNUSED, GVariant * parameters, gpointer user_data)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s.", "dbus signal that key has changed", g_variant_print (parameters, FALSE));
GVariant * variant = g_variant_get_child_value (parameters, 0);
const gchar * keypathname = g_variant_get_string (variant, NULL);
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) user_data;
GElektraKeySet * gks_keys = esb->subscription_gks_keys;
GElektraKeySet * gks_paths = esb->subscription_gks_paths;
const gchar * gsettingspath = g_strdup (g_strstr_len (g_strstr_len (keypathname, -1, "/") + 1, -1, "/"));
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s", "keypathname: ", keypathname);
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s", "GSEttings Path: ", gsettingspath);
// we do not expect paths here
g_assert (!g_str_has_suffix (keypathname, "/"));
GElektraKey * gkey = gelektra_keyset_lookup_byname (gks_keys, keypathname, GELEKTRA_KDB_O_NONE);
int found = 0;
if (gkey)
{
gchar * gsettingskeyname = g_strdup (g_strstr_len (g_strstr_len (gelektra_key_name (gkey), -1, "/") + 1, -1, "/"));
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s: %s", "Subscribed key changed", gsettingskeyname);
g_settings_backend_changed (G_SETTINGS_BACKEND (user_data), gsettingskeyname, NULL);
g_free (gsettingskeyname);
found = 1;
}
else
{
GElektraKey * needle = gelektra_key_new (keypathname, KEY_VALUE, "", KEY_END);
GElektraKey * cur;
gssize pos = 0;
while ((cur = gelektra_keyset_at (gks_paths, pos)) != NULL)
{
if (gelektra_key_isbeloworsame (needle, cur))
{
gchar * gsettingskeyname =
g_strdup (g_strstr_len (g_strstr_len (gelektra_key_name (needle), -1, "/") + 1, -1, "/"));
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s: %s", "Key below subscribed path changed", gsettingskeyname);
g_settings_backend_changed (G_SETTINGS_BACKEND (user_data), gsettingskeyname, NULL);
g_free (gsettingskeyname);
found = 1;
break;
}
pos++;
}
}
if (!found)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s: %s", "Not subscribed to key", keypathname);
}
g_variant_unref (variant);
}
static void elektra_settings_bus_connected (GObject * source_object G_GNUC_UNUSED, GAsyncResult * res, gpointer user_data)
{
GError * err = NULL;
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) user_data;
GDBusConnection * connection = g_bus_get_finish (res, &err);
if (esb->dbus_connections[0] == NULL)
{
esb->dbus_connections[0] = connection;
}
else if (esb->dbus_connections[1] == NULL)
{
esb->dbus_connections[1] = connection;
}
else
return;
if (err != NULL)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s!", "Error on connection to dbus:", err->message);
return;
}
g_dbus_connection_signal_subscribe (connection, NULL, "org.libelektra", NULL, "/org/libelektra/configuration", NULL,
G_DBUS_SIGNAL_FLAGS_NONE, elektra_settings_key_changed, user_data, NULL);
}
static void elektra_settings_check_bus_connection (ElektraSettingsBackend * backend)
{
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
GCancellable * g_cancellable = g_cancellable_new ();
if (esb->dbus_connections[0] == NULL)
{
g_bus_get (G_BUS_TYPE_SESSION, g_cancellable, elektra_settings_bus_connected, backend);
}
if (esb->dbus_connections[1] == NULL)
{
g_bus_get (G_BUS_TYPE_SYSTEM, g_cancellable, elektra_settings_bus_connected, backend);
}
}
/* elektra_settings_backend_subscribe implements g_settings_backend_subscribe:
* @backend: a #GSettingsBackend
* @name: a key or path to subscribe to
*
* Requests that change signals be emitted for events on @name.
*/
static void elektra_settings_backend_subscribe (GSettingsBackend * backend, const gchar * name)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s.", "Subscribe to:", name);
gchar * lookupPath = g_strconcat ("/", G_ELEKTRA_SETTINGS_PATH, name, NULL);
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
GElektraKeySet * ks = 0;
if (g_str_has_suffix (name, "/"))
{
ks = esb->subscription_gks_paths;
}
else
{
ks = esb->subscription_gks_keys;
}
GElektraKey * gkey = gelektra_keyset_lookup_byname (ks, lookupPath, GELEKTRA_KDB_O_NONE);
if (gkey != NULL)
{
(*(guint *) gelektra_key_getvalue (gkey))++; // TODO: violation of the C API
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s", "Key is already subscribed, incrementing subscription count.");
return;
}
g_free (lookupPath);
guint counter = 1;
gchar * pathToSubscribe = g_strconcat (G_ELEKTRA_SETTINGS_USER, G_ELEKTRA_SETTINGS_PATH, name, NULL);
gkey = gelektra_key_new (pathToSubscribe, KEY_BINARY, KEY_SIZE, sizeof (guint), // now the size is important
KEY_VALUE, &counter, // sets the binary value of the counter
KEY_END);
g_free (pathToSubscribe);
if (gelektra_keyset_append (ks, gkey) == -1)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Could not append the key to subscription keyset!");
return;
}
}
/* elektra_settings_backend_unsubscribe implements g_settings_backend_unsubscribe:
* @backend: a #GSettingsBackend
* @name: a key or path to subscribe to
*
* Reverses the effect of a previous call to
* g_settings_backend_subscribe().
*/
static void elektra_settings_backend_unsubscribe (GSettingsBackend * backend, const gchar * name)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s %s.", "Unsubscribe:", name);
gchar * lookupPath = g_strconcat ("/", G_ELEKTRA_SETTINGS_PATH, name, NULL);
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) backend;
GElektraKeySet * ks = 0;
if (g_str_has_suffix (name, "/"))
{
ks = esb->subscription_gks_paths;
}
else
{
ks = esb->subscription_gks_keys;
}
GElektraKey * gkey = gelektra_keyset_lookup_byname (ks, lookupPath, GELEKTRA_KDB_O_NONE);
// TODO CHECK VALUE BEFORE working with it
if (gkey != NULL)
{
guint * counter = (guint *) gelektra_key_getvalue (gkey);
(*counter)--;
if (*counter == 0)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s", "Subscription found deleting");
gelektra_keyset_lookup (ks, gkey, GELEKTRA_KDB_O_POP);
}
g_free (lookupPath);
return;
}
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s", "Subscription not found");
g_free (lookupPath);
return;
}
/*
* Open elektra on empty level, as we asume that any application using GSettings is
* an application and GSettings itself does not offer any such distinction. In addition
* we don't know the application path in advance.
*/
static void elektra_settings_backend_init (ElektraSettingsBackend * esb)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Init new ElektraSettingsBackend");
esb->gkey_error = gelektra_key_new (0);
esb->gkey_user = gelektra_key_new (G_ELEKTRA_SETTINGS_USER G_ELEKTRA_SETTINGS_PATH, KEY_END);
esb->gkey_system = gelektra_key_new (G_ELEKTRA_SETTINGS_SYSTEM G_ELEKTRA_SETTINGS_PATH, KEY_END);
esb->gkdb = gelektra_kdb_open (NULL, esb->gkey_error);
esb->gks_user = gelektra_keyset_new (0, GELEKTRA_KEYSET_END);
esb->gks_system = gelektra_keyset_new (0, GELEKTRA_KEYSET_END);
esb->subscription_gks_keys = gelektra_keyset_new (0, GELEKTRA_KEYSET_END);
esb->subscription_gks_paths = gelektra_keyset_new (0, GELEKTRA_KEYSET_END);
gelektra_kdb_get (esb->gkdb, esb->gks_user, esb->gkey_user);
gelektra_kdb_get (esb->gkdb, esb->gks_system, esb->gkey_system);
elektra_settings_check_bus_connection (esb);
}
/*
* Cleanup
*/
static void elektra_settings_backend_finalize (GObject * object)
{
g_log (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "%s.", "Finalize ElektraSettingsBackend");
ElektraSettingsBackend * esb = (ElektraSettingsBackend *) object;
gelektra_kdb_close (esb->gkdb, esb->gkey_error);
// TODO error handling
G_OBJECT_CLASS (elektra_settings_backend_parent_class)->finalize (object);
}
static void elektra_settings_backend_class_init (GSettingsBackendClass * class)
{
GObjectClass * object_class = G_OBJECT_CLASS (class);
object_class->finalize = elektra_settings_backend_finalize;
class->read = elektra_settings_backend_read;
class->read_user_value = elektra_settings_backend_read_user_value;
class->write = elektra_settings_backend_write;
class->write_tree = elektra_settings_backend_write_tree;
class->reset = elektra_settings_backend_reset;
class->get_writable = elektra_settings_backend_get_writable;
class->subscribe = elektra_settings_backend_subscribe;
class->unsubscribe = elektra_settings_backend_unsubscribe;
class->sync = elektra_settings_backend_sync;
}
void g_io_module_load (GIOModule * module)
{
g_type_module_use (G_TYPE_MODULE (module));
g_io_extension_point_implement (G_SETTINGS_BACKEND_EXTENSION_POINT_NAME, elektra_settings_backend_get_type (), "elektra",
G_ELEKTRA_SETTINGS_MODULE_PRIORITY);
}
void g_io_module_unload (GIOModule * module G_GNUC_UNUSED)
{
g_assert_not_reached ();
}
gchar ** g_io_module_query (void)
{
return g_strsplit (G_SETTINGS_BACKEND_EXTENSION_POINT_NAME, "!", 0);
}
#undef G_LOG_DOMAIN