| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | /*
 | 
					
						
							|  |  |  |  |  * Copyright © 2010 Codethink Limited | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2022-05-18 09:12:45 +01:00
										 |  |  |  |  * SPDX-License-Identifier: LGPL-2.1-or-later | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |  * This library is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							| 
									
										
										
										
											2017-05-27 18:21:30 +02:00
										 |  |  |  |  * version 2.1 of the License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * This library 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 | 
					
						
							|  |  |  |  |  * Lesser General Public License for more details. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * You should have received a copy of the GNU Lesser General Public | 
					
						
							| 
									
										
										
										
											2014-01-23 12:58:29 +01:00
										 |  |  |  |  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Author: Ryan Lortie <desrt@desrt.ca> | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | /* Prologue {{{1 */ | 
					
						
							| 
									
										
										
										
											2010-04-20 22:41:12 -04:00
										 |  |  |  | #include "config.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-23 16:49:21 +06:00
										 |  |  |  | #include <glib.h>
 | 
					
						
							| 
									
										
										
										
											2010-05-19 16:02:05 -04:00
										 |  |  |  | #include <gstdio.h>
 | 
					
						
							| 
									
										
										
										
											2011-04-11 12:57:19 -04:00
										 |  |  |  | #include <gi18n.h>
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | #include <string.h>
 | 
					
						
							|  |  |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2011-04-11 12:57:19 -04:00
										 |  |  |  | #include <locale.h>
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "gvdb/gvdb-builder.h"
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | #include "strinfo.c"
 | 
					
						
							| 
									
										
										
										
											2012-11-10 10:51:18 -05:00
										 |  |  |  | #include "glib/glib-private.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-28 01:09:08 +01:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | strip_string (GString *string) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gint i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   for (i = 0; g_ascii_isspace (string->str[i]); i++); | 
					
						
							|  |  |  |  |   g_string_erase (string, 0, i); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (string->len > 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       /* len > 0, so there must be at least one non-whitespace character */ | 
					
						
							|  |  |  |  |       for (i = string->len - 1; g_ascii_isspace (string->str[i]); i--); | 
					
						
							|  |  |  |  |       g_string_truncate (string, i + 1); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | /* Handling of <enum> {{{1 */ | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  | typedef struct | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GString *strinfo; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   gboolean is_flags; | 
					
						
							|  |  |  |  | } EnumState; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | enum_state_free (gpointer data) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   EnumState *state = data; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   g_string_free (state->strinfo, TRUE); | 
					
						
							|  |  |  |  |   g_slice_free (EnumState, state); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-16 21:53:03 -04:00
										 |  |  |  | static EnumState * | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  | enum_state_new (gboolean is_flags) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   EnumState *state; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state = g_slice_new (EnumState); | 
					
						
							|  |  |  |  |   state->strinfo = g_string_new (NULL); | 
					
						
							|  |  |  |  |   state->is_flags = is_flags; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return state; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | enum_state_add_value (EnumState    *state, | 
					
						
							|  |  |  |  |                       const gchar  *nick, | 
					
						
							|  |  |  |  |                       const gchar  *valuestr, | 
					
						
							|  |  |  |  |                       GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gint64 value; | 
					
						
							|  |  |  |  |   gchar *end; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (nick[0] == '\0' || nick[1] == '\0') | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                    _("nick must be a minimum of 2 characters")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   value = g_ascii_strtoll (valuestr, &end, 0); | 
					
						
							| 
									
										
										
										
											2015-01-15 14:33:17 -08:00
										 |  |  |  |   if (*end || (state->is_flags ? | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                 (value > G_MAXUINT32 || value < 0) : | 
					
						
							| 
									
										
										
										
											2015-01-15 14:33:17 -08:00
										 |  |  |  |                 (value > G_MAXINT32 || value < G_MININT32))) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                    _("Invalid numeric value")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   if (strinfo_builder_contains (state->strinfo, nick)) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                    _("<value nick='%s'/> already specified"), nick); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   if (strinfo_builder_contains_value (state->strinfo, value)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                    _("value='%s' already specified"), valuestr); | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-04 02:57:06 -04:00
										 |  |  |  |   /* Silently drop the null case if it is mentioned.
 | 
					
						
							|  |  |  |  |    * It is properly denoted with an empty array. | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   if (state->is_flags && value == 0) | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   if (state->is_flags && (value & (value - 1))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                    _("flags values must have at most 1 bit set")); | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* Since we reject exact duplicates of value='' and we only allow one
 | 
					
						
							|  |  |  |  |    * bit to be set, it's not possible to have overlaps. | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * If we loosen the one-bit-set restriction we need an overlap check. | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   strinfo_builder_append_item (state->strinfo, nick, value); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | enum_state_end (EnumState **state_ptr, | 
					
						
							|  |  |  |  |                 GError    **error) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   EnumState *state; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   state = *state_ptr; | 
					
						
							|  |  |  |  |   *state_ptr = NULL; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   if (state->strinfo->len == 0) | 
					
						
							| 
									
										
										
										
											2010-09-27 10:36:11 -04:00
										 |  |  |  |     g_set_error (error, | 
					
						
							|  |  |  |  |                  G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                  _("<%s> must contain at least one <value>"), | 
					
						
							| 
									
										
										
										
											2010-09-27 10:36:11 -04:00
										 |  |  |  |                  state->is_flags ? "flags" : "enum"); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Handling of <key> {{{1 */ | 
					
						
							|  |  |  |  | typedef struct | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   /* for <child>, @child_schema will be set.
 | 
					
						
							|  |  |  |  |    * for <key>, everything else will be set. | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   gchar        *child_schema; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   GVariantType *type; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   gboolean      have_gettext_domain; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   gchar         l10n; | 
					
						
							|  |  |  |  |   gchar        *l10n_context; | 
					
						
							|  |  |  |  |   GString      *unparsed_default_value; | 
					
						
							|  |  |  |  |   GVariant     *default_value; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |   GVariantDict *desktop_overrides; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   GString      *strinfo; | 
					
						
							|  |  |  |  |   gboolean      is_enum; | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   gboolean      is_flags; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   GVariant     *minimum; | 
					
						
							|  |  |  |  |   GVariant     *maximum; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   gboolean      has_choices; | 
					
						
							|  |  |  |  |   gboolean      has_aliases; | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |   gboolean      is_override; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   gboolean      checked; | 
					
						
							|  |  |  |  |   GVariant     *serialised; | 
					
						
							| 
									
										
										
										
											2015-04-01 18:55:54 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   gboolean      summary_seen; | 
					
						
							|  |  |  |  |   gboolean      description_seen; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | } KeyState; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static KeyState * | 
					
						
							|  |  |  |  | key_state_new (const gchar *type_string, | 
					
						
							|  |  |  |  |                const gchar *gettext_domain, | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                gboolean     is_enum, | 
					
						
							|  |  |  |  |                gboolean     is_flags, | 
					
						
							|  |  |  |  |                GString     *strinfo) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   KeyState *state; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state = g_slice_new0 (KeyState); | 
					
						
							|  |  |  |  |   state->type = g_variant_type_new (type_string); | 
					
						
							|  |  |  |  |   state->have_gettext_domain = gettext_domain != NULL; | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   state->is_enum = is_enum; | 
					
						
							|  |  |  |  |   state->is_flags = is_flags; | 
					
						
							| 
									
										
										
										
											2015-04-01 18:55:54 -04:00
										 |  |  |  |   state->summary_seen = FALSE; | 
					
						
							|  |  |  |  |   state->description_seen = FALSE; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   if (strinfo) | 
					
						
							|  |  |  |  |     state->strinfo = g_string_new_len (strinfo->str, strinfo->len); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   else | 
					
						
							|  |  |  |  |     state->strinfo = g_string_new (NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return state; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  | static KeyState * | 
					
						
							|  |  |  |  | key_state_override (KeyState    *state, | 
					
						
							|  |  |  |  |                     const gchar *gettext_domain) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   KeyState *copy; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   copy = g_slice_new0 (KeyState); | 
					
						
							|  |  |  |  |   copy->type = g_variant_type_copy (state->type); | 
					
						
							|  |  |  |  |   copy->have_gettext_domain = gettext_domain != NULL; | 
					
						
							|  |  |  |  |   copy->strinfo = g_string_new_len (state->strinfo->str, | 
					
						
							|  |  |  |  |                                     state->strinfo->len); | 
					
						
							|  |  |  |  |   copy->is_enum = state->is_enum; | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   copy->is_flags = state->is_flags; | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |   copy->is_override = TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->minimum) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       copy->minimum = g_variant_ref (state->minimum); | 
					
						
							|  |  |  |  |       copy->maximum = g_variant_ref (state->maximum); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return copy; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | static KeyState * | 
					
						
							|  |  |  |  | key_state_new_child (const gchar *child_schema) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   KeyState *state; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state = g_slice_new0 (KeyState); | 
					
						
							|  |  |  |  |   state->child_schema = g_strdup (child_schema); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return state; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							|  |  |  |  | is_valid_choices (GVariant *variant, | 
					
						
							|  |  |  |  |                   GString  *strinfo) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   switch (g_variant_classify (variant)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       case G_VARIANT_CLASS_MAYBE: | 
					
						
							|  |  |  |  |       case G_VARIANT_CLASS_ARRAY: | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           gboolean valid = TRUE; | 
					
						
							|  |  |  |  |           GVariantIter iter; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           g_variant_iter_init (&iter, variant); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           while (valid && (variant = g_variant_iter_next_value (&iter))) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               valid = is_valid_choices (variant, strinfo); | 
					
						
							|  |  |  |  |               g_variant_unref (variant); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           return valid; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case G_VARIANT_CLASS_STRING: | 
					
						
							|  |  |  |  |         return strinfo_is_string_valid ((const guint32 *) strinfo->str, | 
					
						
							|  |  |  |  |                                         strinfo->len / 4, | 
					
						
							|  |  |  |  |                                         g_variant_get_string (variant, NULL)); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       default: | 
					
						
							|  |  |  |  |         g_assert_not_reached (); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Gets called at </default> </choices> or <range/> to check for
 | 
					
						
							|  |  |  |  |  * validity of the default value so that any inconsistency is | 
					
						
							|  |  |  |  |  * reported as soon as it is encountered. | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_check_range (KeyState  *state, | 
					
						
							|  |  |  |  |                        GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (state->default_value) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |       const gchar *tag; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       tag = state->is_override ? "override" : "default"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       if (state->minimum) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           if (g_variant_compare (state->default_value, state->minimum) < 0 || | 
					
						
							|  |  |  |  |               g_variant_compare (state->default_value, state->maximum) > 0) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                            _("<%s> is not contained in " | 
					
						
							|  |  |  |  |                            "the specified range"), tag); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       else if (state->strinfo->len) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           if (!is_valid_choices (state->default_value, state->strinfo)) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               if (state->is_enum) | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |                 g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                              G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                              _("<%s> is not a valid member of " | 
					
						
							|  |  |  |  |                              "the specified enumerated type"), tag); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |               else if (state->is_flags) | 
					
						
							|  |  |  |  |                 g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                              G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                              _("<%s> contains string not in the " | 
					
						
							|  |  |  |  |                              "specified flags type"), tag); | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |               else | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |                 g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                              G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                              _("<%s> contains a string not in " | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                              "<choices>"), tag); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_set_range (KeyState     *state, | 
					
						
							|  |  |  |  |                      const gchar  *min_str, | 
					
						
							|  |  |  |  |                      const gchar  *max_str, | 
					
						
							|  |  |  |  |                      GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-31 15:10:25 +05:30
										 |  |  |  |   const struct { | 
					
						
							|  |  |  |  |     const gchar  type; | 
					
						
							|  |  |  |  |     const gchar *min; | 
					
						
							|  |  |  |  |     const gchar *max; | 
					
						
							|  |  |  |  |   } table[] = { | 
					
						
							|  |  |  |  |     { 'y',                    "0",                  "255" }, | 
					
						
							|  |  |  |  |     { 'n',               "-32768",                "32767" }, | 
					
						
							|  |  |  |  |     { 'q',                    "0",                "65535" }, | 
					
						
							|  |  |  |  |     { 'i',          "-2147483648",           "2147483647" }, | 
					
						
							|  |  |  |  |     { 'u',                    "0",           "4294967295" }, | 
					
						
							|  |  |  |  |     { 'x', "-9223372036854775808",  "9223372036854775807" }, | 
					
						
							|  |  |  |  |     { 't',                    "0", "18446744073709551615" }, | 
					
						
							|  |  |  |  |     { 'd',                 "-inf",                  "inf" }, | 
					
						
							|  |  |  |  |   }; | 
					
						
							|  |  |  |  |   gboolean type_ok = FALSE; | 
					
						
							| 
									
										
										
										
											2020-11-17 22:14:26 +01:00
										 |  |  |  |   gsize i; | 
					
						
							| 
									
										
										
										
											2011-03-31 15:10:25 +05:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   if (state->minimum) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                            _("<range/> already specified for this key")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-31 15:10:25 +05:30
										 |  |  |  |   for (i = 0; i < G_N_ELEMENTS (table); i++) | 
					
						
							|  |  |  |  |     if (*(char *) state->type == table[i].type) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         min_str = min_str ? min_str : table[i].min; | 
					
						
							|  |  |  |  |         max_str = max_str ? max_str : table[i].max; | 
					
						
							|  |  |  |  |         type_ok = TRUE; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!type_ok) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       gchar *type = g_variant_type_dup_string (state->type); | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                   G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                   _("<range> not allowed for keys of type “%s”"), type); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       g_free (type); | 
					
						
							|  |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state->minimum = g_variant_parse (state->type, min_str, NULL, NULL, error); | 
					
						
							|  |  |  |  |   if (state->minimum == NULL) | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state->maximum = g_variant_parse (state->type, max_str, NULL, NULL, error); | 
					
						
							|  |  |  |  |   if (state->maximum == NULL) | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_variant_compare (state->minimum, state->maximum) > 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:30:20 +01:00
										 |  |  |  |                    _("<range> specified minimum is greater than maximum")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   key_state_check_range (state, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static GString * | 
					
						
							|  |  |  |  | key_state_start_default (KeyState     *state, | 
					
						
							|  |  |  |  |                          const gchar  *l10n, | 
					
						
							|  |  |  |  |                          const gchar  *context, | 
					
						
							|  |  |  |  |                          GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (l10n != NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (l10n, "messages") == 0) | 
					
						
							|  |  |  |  |         state->l10n = 'm'; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       else if (strcmp (l10n, "time") == 0) | 
					
						
							|  |  |  |  |         state->l10n = 't'; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                        G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                        _("unsupported l10n category: %s"), l10n); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           return NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (!state->have_gettext_domain) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                                G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                                _("l10n requested, but no " | 
					
						
							|  |  |  |  |                                "gettext domain given")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           return NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       state->l10n_context = g_strdup (context); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   else if (context != NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                            _("translation context given for " | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                            "value without l10n enabled")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_string_new (NULL); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_end_default (KeyState  *state, | 
					
						
							|  |  |  |  |                        GString  **string, | 
					
						
							|  |  |  |  |                        GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   state->unparsed_default_value = *string; | 
					
						
							|  |  |  |  |   *string = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state->default_value = g_variant_parse (state->type, | 
					
						
							|  |  |  |  |                                           state->unparsed_default_value->str, | 
					
						
							|  |  |  |  |                                           NULL, NULL, error); | 
					
						
							| 
									
										
										
										
											2015-05-15 22:53:41 -04:00
										 |  |  |  |   if (!state->default_value) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       gchar *type = g_variant_type_dup_string (state->type); | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |       g_prefix_error (error, _("Failed to parse <default> value of type “%s”: "), type); | 
					
						
							| 
									
										
										
										
											2015-05-15 22:53:41 -04:00
										 |  |  |  |       g_free (type); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   key_state_check_range (state, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_start_choices (KeyState  *state, | 
					
						
							|  |  |  |  |                          GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   const GVariantType *type = state->type; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->is_enum) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                            _("<choices> cannot be specified for keys " | 
					
						
							|  |  |  |  |                            "tagged as having an enumerated type")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->has_choices) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                            _("<choices> already specified for this key")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   while (g_variant_type_is_maybe (type) || g_variant_type_is_array (type)) | 
					
						
							|  |  |  |  |     type = g_variant_type_element (type); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!g_variant_type_equal (type, G_VARIANT_TYPE_STRING)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       gchar *type_string = g_variant_type_dup_string (state->type); | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                    _("<choices> not allowed for keys of type “%s”"), | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                    type_string); | 
					
						
							|  |  |  |  |       g_free (type_string); | 
					
						
							|  |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_add_choice (KeyState     *state, | 
					
						
							|  |  |  |  |                       const gchar  *choice, | 
					
						
							|  |  |  |  |                       GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (strinfo_builder_contains (state->strinfo, choice)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                    _("<choice value='%s'/> already given"), choice); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   strinfo_builder_append_item (state->strinfo, choice, 0); | 
					
						
							|  |  |  |  |   state->has_choices = TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_end_choices (KeyState  *state, | 
					
						
							|  |  |  |  |                        GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!state->has_choices) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                    _("<choices> must contain at least one <choice>")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   key_state_check_range (state, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_start_aliases (KeyState  *state, | 
					
						
							|  |  |  |  |                          GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (state->has_aliases) | 
					
						
							|  |  |  |  |     g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                          G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                          _("<aliases> already specified for this key")); | 
					
						
							| 
									
										
										
										
											2010-10-16 23:30:30 -04:00
										 |  |  |  |   else if (!state->is_flags && !state->is_enum && !state->has_choices) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                          G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                          _("<aliases> can only be specified for keys with " | 
					
						
							|  |  |  |  |                          "enumerated or flags types or after <choices>")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_add_alias (KeyState     *state, | 
					
						
							|  |  |  |  |                      const gchar  *alias, | 
					
						
							|  |  |  |  |                      const gchar  *target, | 
					
						
							|  |  |  |  |                      GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (strinfo_builder_contains (state->strinfo, alias)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strinfo_is_string_valid ((guint32 *) state->strinfo->str, | 
					
						
							|  |  |  |  |                                    state->strinfo->len / 4, | 
					
						
							|  |  |  |  |                                    alias)) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           if (state->is_enum) | 
					
						
							|  |  |  |  |             g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                          G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                          _("<alias value='%s'/> given when “%s” is already " | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                          "a member of the enumerated type"), alias, alias); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |           else | 
					
						
							|  |  |  |  |             g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                          G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                          _("<alias value='%s'/> given when " | 
					
						
							|  |  |  |  |                          "<choice value='%s'/> was already given"), | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                          alias, alias); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                      G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                      _("<alias value='%s'/> already specified"), alias); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!strinfo_builder_append_alias (state->strinfo, alias, target)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                    state->is_enum ? | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                      _("alias target “%s” is not in enumerated type") : | 
					
						
							|  |  |  |  |                      _("alias target “%s” is not in <choices>"), | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                    target); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state->has_aliases = TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_end_aliases (KeyState  *state, | 
					
						
							|  |  |  |  |                        GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!state->has_aliases) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-02 17:19:30 +02:00
										 |  |  |  |                    _("<aliases> must contain at least one <alias>")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							|  |  |  |  | key_state_check (KeyState  *state, | 
					
						
							|  |  |  |  |                  GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (state->checked) | 
					
						
							|  |  |  |  |     return TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return state->checked = TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static GVariant * | 
					
						
							|  |  |  |  | key_state_serialise (KeyState *state) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (state->serialised == NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (state->child_schema) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           state->serialised = g_variant_new_string (state->child_schema); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           GVariantBuilder builder; | 
					
						
							| 
									
										
										
										
											2019-03-05 12:22:51 +00:00
										 |  |  |  |           gboolean checked G_GNUC_UNUSED  /* when compiling with G_DISABLE_ASSERT */; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 12:22:51 +00:00
										 |  |  |  |           checked = key_state_check (state, NULL); | 
					
						
							|  |  |  |  |           g_assert (checked); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-25 11:09:07 -07:00
										 |  |  |  |           g_variant_builder_init_static (&builder, G_VARIANT_TYPE_TUPLE); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |           /* default value */ | 
					
						
							|  |  |  |  |           g_variant_builder_add_value (&builder, state->default_value); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           /* translation */ | 
					
						
							|  |  |  |  |           if (state->l10n) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2012-01-28 01:09:08 +01:00
										 |  |  |  |               /* We are going to store the untranslated default for
 | 
					
						
							|  |  |  |  |                * runtime translation according to the current locale. | 
					
						
							|  |  |  |  |                * We need to strip leading and trailing whitespace from | 
					
						
							|  |  |  |  |                * the string so that it's exactly the same as the one | 
					
						
							|  |  |  |  |                * that ended up in the .po file for translation. | 
					
						
							|  |  |  |  |                * | 
					
						
							|  |  |  |  |                * We want to do this so that | 
					
						
							|  |  |  |  |                * | 
					
						
							|  |  |  |  |                *   <default l10n='messages'> | 
					
						
							|  |  |  |  |                *     ['a', 'b', 'c'] | 
					
						
							|  |  |  |  |                *   </default> | 
					
						
							|  |  |  |  |                * | 
					
						
							|  |  |  |  |                * ends up in the .po file like "['a', 'b', 'c']", | 
					
						
							|  |  |  |  |                * omitting the extra whitespace at the start and end. | 
					
						
							|  |  |  |  |                */ | 
					
						
							|  |  |  |  |               strip_string (state->unparsed_default_value); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |               if (state->l10n_context) | 
					
						
							|  |  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2024-04-25 00:37:47 +01:00
										 |  |  |  |                   size_t len; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |                   /* Contextified messages are supported by prepending
 | 
					
						
							|  |  |  |  |                    * the context, followed by '\004' to the start of the | 
					
						
							|  |  |  |  |                    * message string.  We do that here to save GSettings | 
					
						
							|  |  |  |  |                    * the work later on. | 
					
						
							|  |  |  |  |                    */ | 
					
						
							|  |  |  |  |                   len = strlen (state->l10n_context); | 
					
						
							|  |  |  |  |                   state->l10n_context[len] = '\004'; | 
					
						
							|  |  |  |  |                   g_string_prepend_len (state->unparsed_default_value, | 
					
						
							|  |  |  |  |                                         state->l10n_context, len + 1); | 
					
						
							|  |  |  |  |                   g_free (state->l10n_context); | 
					
						
							|  |  |  |  |                   state->l10n_context = NULL; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               g_variant_builder_add (&builder, "(y(y&s))", 'l', state->l10n, | 
					
						
							| 
									
										
										
										
											2010-08-03 14:32:29 -04:00
										 |  |  |  |                                      state->unparsed_default_value->str); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |               g_string_free (state->unparsed_default_value, TRUE); | 
					
						
							|  |  |  |  |               state->unparsed_default_value = NULL; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           /* choice, aliases, enums */ | 
					
						
							|  |  |  |  |           if (state->strinfo->len) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               GVariant *array; | 
					
						
							| 
									
										
										
										
											2010-10-03 23:26:18 -04:00
										 |  |  |  |               guint32 *words; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |               gpointer data; | 
					
						
							|  |  |  |  |               gsize size; | 
					
						
							| 
									
										
										
										
											2020-11-17 22:14:26 +01:00
										 |  |  |  |               gsize i; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |               size = state->strinfo->len; | 
					
						
							| 
									
										
										
										
											2023-01-24 19:07:46 +03:00
										 |  |  |  |               data = g_string_free_and_steal (g_steal_pointer (&state->strinfo)); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-03 23:26:18 -04:00
										 |  |  |  |               words = data; | 
					
						
							|  |  |  |  |               for (i = 0; i < size / sizeof (guint32); i++) | 
					
						
							|  |  |  |  |                 words[i] = GUINT32_TO_LE (words[i]); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |               array = g_variant_new_from_data (G_VARIANT_TYPE ("au"), | 
					
						
							|  |  |  |  |                                                data, size, TRUE, | 
					
						
							|  |  |  |  |                                                g_free, data); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               g_variant_builder_add (&builder, "(y@au)", | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                                      state->is_flags ? 'f' : | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                                      state->is_enum ? 'e' : 'c', | 
					
						
							|  |  |  |  |                                      array); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           /* range */ | 
					
						
							|  |  |  |  |           if (state->minimum || state->maximum) | 
					
						
							|  |  |  |  |             g_variant_builder_add (&builder, "(y(**))", 'r', | 
					
						
							|  |  |  |  |                                    state->minimum, state->maximum); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |           /* per-desktop overrides */ | 
					
						
							|  |  |  |  |           if (state->desktop_overrides) | 
					
						
							|  |  |  |  |             g_variant_builder_add (&builder, "(y@a{sv})", 'd', | 
					
						
							|  |  |  |  |                                    g_variant_dict_end (state->desktop_overrides)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           state->serialised = g_variant_builder_end (&builder); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       g_variant_ref_sink (state->serialised); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_variant_ref (state->serialised); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_free (gpointer data) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   KeyState *state = data; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-11 11:03:03 +01:00
										 |  |  |  |   g_free (state->child_schema); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   if (state->type) | 
					
						
							|  |  |  |  |     g_variant_type_free (state->type); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_free (state->l10n_context); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->unparsed_default_value) | 
					
						
							|  |  |  |  |     g_string_free (state->unparsed_default_value, TRUE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->default_value) | 
					
						
							|  |  |  |  |     g_variant_unref (state->default_value); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->strinfo) | 
					
						
							|  |  |  |  |     g_string_free (state->strinfo, TRUE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->minimum) | 
					
						
							|  |  |  |  |     g_variant_unref (state->minimum); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->maximum) | 
					
						
							|  |  |  |  |     g_variant_unref (state->maximum); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->serialised) | 
					
						
							|  |  |  |  |     g_variant_unref (state->serialised); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |   if (state->desktop_overrides) | 
					
						
							|  |  |  |  |     g_variant_dict_unref (state->desktop_overrides); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   g_slice_free (KeyState, state); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Key name validity {{{1 */ | 
					
						
							| 
									
										
										
										
											2010-04-21 09:33:57 -04:00
										 |  |  |  | static gboolean allow_any_name = FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | static gboolean | 
					
						
							|  |  |  |  | is_valid_keyname (const gchar  *key, | 
					
						
							|  |  |  |  |                   GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gint i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (key[0] == '\0') | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-04-19 14:18:59 -04:00
										 |  |  |  |       g_set_error_literal (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                            _("Empty names are not permitted")); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-21 09:33:57 -04:00
										 |  |  |  |   if (allow_any_name) | 
					
						
							|  |  |  |  |     return TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   if (!g_ascii_islower (key[0])) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                    _("Invalid name “%s”: names must begin " | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                      "with a lowercase letter"), key); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   for (i = 1; key[i]; i++) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (key[i] != '-' && | 
					
						
							|  |  |  |  |           !g_ascii_islower (key[i]) && | 
					
						
							|  |  |  |  |           !g_ascii_isdigit (key[i])) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                        _("Invalid name “%s”: invalid character “%c”; " | 
					
						
							|  |  |  |  |                          "only lowercase letters, numbers and hyphen (“-”) " | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                          "are permitted"), key, key[i]); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return FALSE; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (key[i] == '-' && key[i + 1] == '-') | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                        _("Invalid name “%s”: two successive hyphens (“--”) " | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                          "are not permitted"), key); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return FALSE; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (key[i - 1] == '-') | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                    _("Invalid name “%s”: the last character may not be a " | 
					
						
							|  |  |  |  |                      "hyphen (“-”)"), key); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-19 16:12:30 +02:00
										 |  |  |  |   if (i > 1024) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                    _("Invalid name “%s”: maximum length is 1024"), key); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | /* Handling of <schema> {{{1 */ | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  | typedef struct _SchemaState SchemaState; | 
					
						
							|  |  |  |  | struct _SchemaState | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |   SchemaState *extends; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   gchar       *path; | 
					
						
							|  |  |  |  |   gchar       *gettext_domain; | 
					
						
							|  |  |  |  |   gchar       *extends_name; | 
					
						
							|  |  |  |  |   gchar       *list_of; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |   GHashTable  *keys; | 
					
						
							|  |  |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static SchemaState * | 
					
						
							|  |  |  |  | schema_state_new (const gchar  *path, | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |                   const gchar  *gettext_domain, | 
					
						
							|  |  |  |  |                   SchemaState  *extends, | 
					
						
							|  |  |  |  |                   const gchar  *extends_name, | 
					
						
							|  |  |  |  |                   const gchar  *list_of) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   SchemaState *state; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state = g_slice_new (SchemaState); | 
					
						
							|  |  |  |  |   state->path = g_strdup (path); | 
					
						
							|  |  |  |  |   state->gettext_domain = g_strdup (gettext_domain); | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |   state->extends = extends; | 
					
						
							|  |  |  |  |   state->extends_name = g_strdup (extends_name); | 
					
						
							|  |  |  |  |   state->list_of = g_strdup (list_of); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   state->keys = g_hash_table_new_full (g_str_hash, g_str_equal, | 
					
						
							|  |  |  |  |                                        g_free, key_state_free); | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   return state; | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | schema_state_free (gpointer data) | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   SchemaState *state = data; | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   g_free (state->path); | 
					
						
							|  |  |  |  |   g_free (state->gettext_domain); | 
					
						
							| 
									
										
										
										
											2018-04-11 11:03:03 +01:00
										 |  |  |  |   g_free (state->extends_name); | 
					
						
							|  |  |  |  |   g_free (state->list_of); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   g_hash_table_unref (state->keys); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   g_slice_free (SchemaState, state); | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | schema_state_add_child (SchemaState  *state, | 
					
						
							|  |  |  |  |                         const gchar  *name, | 
					
						
							|  |  |  |  |                         const gchar  *schema, | 
					
						
							|  |  |  |  |                         GError      **error) | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   gchar *childname; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!is_valid_keyname (name, error)) | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   childname = g_strconcat (name, "/", NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_hash_table_lookup (state->keys, childname)) | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                    _("<child name='%s'> already specified"), name); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   g_hash_table_insert (state->keys, childname, | 
					
						
							|  |  |  |  |                        key_state_new_child (schema)); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | static KeyState * | 
					
						
							|  |  |  |  | schema_state_add_key (SchemaState  *state, | 
					
						
							|  |  |  |  |                       GHashTable   *enum_table, | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                       GHashTable   *flags_table, | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                       const gchar  *name, | 
					
						
							|  |  |  |  |                       const gchar  *type_string, | 
					
						
							|  |  |  |  |                       const gchar  *enum_type, | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                       const gchar  *flags_type, | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                       GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |   SchemaState *node; | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   GString *strinfo; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   KeyState *key; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |   if (state->list_of) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                            _("Cannot add keys to a “list-of” schema")); | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |       return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   if (!is_valid_keyname (name, error)) | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_hash_table_lookup (state->keys, name)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                    _("<key name='%s'> already specified"), name); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |   for (node = state; node; node = node->extends) | 
					
						
							|  |  |  |  |     if (node->extends) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         KeyState *shadow; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         shadow = g_hash_table_lookup (node->extends->keys, name); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /* in case of <key> <override> <key> make sure we report the
 | 
					
						
							|  |  |  |  |          * location of the original <key>, not the <override>. | 
					
						
							|  |  |  |  |          */ | 
					
						
							|  |  |  |  |         if (shadow && !shadow->is_override) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                          G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                          _("<key name='%s'> shadows <key name='%s'> in " | 
					
						
							|  |  |  |  |                            "<schema id='%s'>; use <override> to modify value"), | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |                          name, name, node->extends_name); | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   if ((type_string != NULL) + (enum_type != NULL) + (flags_type != NULL) != 1) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_MISSING_ATTRIBUTE, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                    _("Exactly one of “type”, “enum” or “flags” must " | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                      "be specified as an attribute to <key>")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   if (type_string == NULL) /* flags or enums was specified */ | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |       EnumState *enum_state; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (enum_type) | 
					
						
							|  |  |  |  |         enum_state = g_hash_table_lookup (enum_table, enum_type); | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         enum_state = g_hash_table_lookup (flags_table, flags_type); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (enum_state == NULL) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                        G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                        _("<%s id='%s'> not (yet) defined."), | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                        flags_type ? "flags"    : "enum", | 
					
						
							|  |  |  |  |                        flags_type ? flags_type : enum_type); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           return NULL; | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |       type_string = flags_type ? "as" : "s"; | 
					
						
							|  |  |  |  |       strinfo = enum_state->strinfo; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (!g_variant_type_string_is_valid (type_string)) | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                        G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                        _("Invalid GVariant type string “%s”"), type_string); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           return NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |       strinfo = NULL; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   key = key_state_new (type_string, state->gettext_domain, | 
					
						
							|  |  |  |  |                        enum_type != NULL, flags_type != NULL, strinfo); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   g_hash_table_insert (state->keys, g_strdup (name), key); | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   return key; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | schema_state_add_override (SchemaState  *state, | 
					
						
							|  |  |  |  |                            KeyState    **key_state, | 
					
						
							|  |  |  |  |                            GString     **string, | 
					
						
							|  |  |  |  |                            const gchar  *key, | 
					
						
							|  |  |  |  |                            const gchar  *l10n, | 
					
						
							|  |  |  |  |                            const gchar  *context, | 
					
						
							|  |  |  |  |                            GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   SchemaState *parent; | 
					
						
							| 
									
										
										
										
											2022-01-19 18:45:20 +01:00
										 |  |  |  |   KeyState *original = NULL; | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->extends == NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error_literal (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                            _("<override> given but schema isn’t " | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                              "extending anything")); | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   for (parent = state->extends; parent; parent = parent->extends) | 
					
						
							|  |  |  |  |     if ((original = g_hash_table_lookup (parent->keys, key))) | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (original == NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                    _("No <key name='%s'> to override"), key); | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_hash_table_lookup (state->keys, key)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                    _("<override name='%s'> already specified"), key); | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   *key_state = key_state_override (original, state->gettext_domain); | 
					
						
							|  |  |  |  |   *string = key_state_start_default (*key_state, l10n, context, error); | 
					
						
							|  |  |  |  |   g_hash_table_insert (state->keys, g_strdup (key), *key_state); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | override_state_end (KeyState **key_state, | 
					
						
							|  |  |  |  |                     GString  **string, | 
					
						
							|  |  |  |  |                     GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   key_state_end_default (*key_state, string, error); | 
					
						
							|  |  |  |  |   *key_state = NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | /* Handling of toplevel state {{{1 */ | 
					
						
							|  |  |  |  | typedef struct | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
											  
											
												gsettings: stay compatible with installed schemas
Bug 747209 introduced an error when multiple <summary> or <description>
tags are found for a single key in a GSettings schema.  This check
should have been present from the start, but it was left out because the
schema compiler doesn't include these items in the cache file.  Even
still -- part of the schema compiler's job is validation, and it should
be enforcing proper syntax here.
Repeated <summary> and <description> tags are a semi-common problem when
intltool has been misconfigured in the build system of a package, but
it's possible to imagine mistakes being made by hand as well.
The idea is that these problems would be caught during the build of a
package and maintainers would be forced to fix their build systems.
An unintended side-effect of this change, however, is that the schema
compiler started ignoring already-installed schemas that contained these
problems, when rebuilding the cache.  This means that the installation
of _any_ application would cause the regeneration of the entire cache,
with these already-installed applications being excluded.  Without the
schema in the cache, the application would crash on next startup.
The validation check in the gsettings m4 macro passes --strict to the
compiler, which is not used when rebuilding the cache after
installation.  Pass this flag down into the parser and only throw the
error in case --strict was given.  This will result in the (desired)
build failure without also causing already-installed apps to stop
functioning.
This means that we will not get even a warning about the invalid schema
file in the already-installed case, but that's fine.  There is no sense
spamming the user with these messages when they are already quite fatal
for the developer at build time.
https://bugzilla.gnome.org/show_bug.cgi?id=747472
											
										 
											2015-04-08 21:55:58 -04:00
										 |  |  |  |   gboolean     strict;                  /* TRUE if --strict was given */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   GHashTable  *schema_table;            /* string -> SchemaState */ | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   GHashTable  *flags_table;             /* string -> EnumState */ | 
					
						
							|  |  |  |  |   GHashTable  *enum_table;              /* string -> EnumState */ | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |   GSList      *this_file_schemas;       /* strings: <schema>s in this file */ | 
					
						
							|  |  |  |  |   GSList      *this_file_flagss;        /* strings: <flags>s in this file */ | 
					
						
							|  |  |  |  |   GSList      *this_file_enums;         /* strings: <enum>s in this file */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   gchar       *schemalist_domain;       /* the <schemalist> gettext domain */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   SchemaState *schema_state;            /* non-NULL when inside <schema> */ | 
					
						
							|  |  |  |  |   KeyState    *key_state;               /* non-NULL when inside <key> */ | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   EnumState   *enum_state;              /* non-NULL when inside <enum> */ | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   GString     *string;                  /* non-NULL when accepting text */ | 
					
						
							|  |  |  |  | } ParseState; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  | static gboolean | 
					
						
							|  |  |  |  | is_subclass (const gchar *class_name, | 
					
						
							|  |  |  |  |              const gchar *possible_parent, | 
					
						
							|  |  |  |  |              GHashTable  *schema_table) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   SchemaState *class; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (strcmp (class_name, possible_parent) == 0) | 
					
						
							|  |  |  |  |     return TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   class = g_hash_table_lookup (schema_table, class_name); | 
					
						
							|  |  |  |  |   g_assert (class != NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return class->extends_name && | 
					
						
							|  |  |  |  |          is_subclass (class->extends_name, possible_parent, schema_table); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | parse_state_start_schema (ParseState  *state, | 
					
						
							|  |  |  |  |                           const gchar  *id, | 
					
						
							|  |  |  |  |                           const gchar  *path, | 
					
						
							|  |  |  |  |                           const gchar  *gettext_domain, | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |                           const gchar  *extends_name, | 
					
						
							|  |  |  |  |                           const gchar  *list_of, | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                           GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |   SchemaState *extends; | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |   gchar *my_id; | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   if (g_hash_table_lookup (state->schema_table, id)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                    _("<schema id='%s'> already specified"), id); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |   if (extends_name) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       extends = g_hash_table_lookup (state->schema_table, extends_name); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (extends == NULL) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                        G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
											  
											
												Rework the build system for a new tests approach
Perform a substantial cleanup of the build system with respect to
building and installing testcases.
First, Makefile.decl has been renamed glib.mk and substantially
expanded.  We intend to add more stuff here in the future, like canned
rules for mkenums, marshallers, resources, etc.
By default, tests are no longer compiled as part of 'make'.  They will
be built when 'make check' is run.  The old behaviour can be obtained
with --enable-always-build-tests.
--disable-modular-tests is gone (because tests are no longer built by
default).  There is no longer any way to cause 'make check' to be a
no-op, but that's not very useful anyway.
A new glibtests.m4 file is introduced.  Along with glib.mk, this
provides for consistent handling of --enable-installed-tests and
--enable-always-build-tests (mentioned above).
Port our various test-installing Makefiles to the new framework.
This patch substantially improves the situation in the toplevel tests/
directory.  Things are now somewhat under control there.  There were
some tests being built that weren't even being run and we run those now.
The long-running GObject performance tests in this directory have been
removed from 'make check' because they take too long.
As an experiment, 'make check' now runs the testcases on win32 builds,
by default.  We can't run them under gtester (since it uses a pipe to
communicate with the subprocess) so just toss them in TESTS.  Most of
them are passing on win32.
Things are not quite done here, but this patch is already a substantial
improvement.  More to come.
											
										 
											2013-05-30 00:07:32 -04:00
										 |  |  |  |                        _("<schema id='%s'> extends not yet existing " | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                          "schema “%s”"), id, extends_name); | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     extends = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (list_of) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-09-09 16:28:18 -04:00
										 |  |  |  |       SchemaState *tmp; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (!(tmp = g_hash_table_lookup (state->schema_table, list_of))) | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                        G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
											  
											
												Rework the build system for a new tests approach
Perform a substantial cleanup of the build system with respect to
building and installing testcases.
First, Makefile.decl has been renamed glib.mk and substantially
expanded.  We intend to add more stuff here in the future, like canned
rules for mkenums, marshallers, resources, etc.
By default, tests are no longer compiled as part of 'make'.  They will
be built when 'make check' is run.  The old behaviour can be obtained
with --enable-always-build-tests.
--disable-modular-tests is gone (because tests are no longer built by
default).  There is no longer any way to cause 'make check' to be a
no-op, but that's not very useful anyway.
A new glibtests.m4 file is introduced.  Along with glib.mk, this
provides for consistent handling of --enable-installed-tests and
--enable-always-build-tests (mentioned above).
Port our various test-installing Makefiles to the new framework.
This patch substantially improves the situation in the toplevel tests/
directory.  Things are now somewhat under control there.  There were
some tests being built that weren't even being run and we run those now.
The long-running GObject performance tests in this directory have been
removed from 'make check' because they take too long.
As an experiment, 'make check' now runs the testcases on win32 builds,
by default.  We can't run them under gtester (since it uses a pipe to
communicate with the subprocess) so just toss them in TESTS.  Most of
them are passing on win32.
Things are not quite done here, but this patch is already a substantial
improvement.  More to come.
											
										 
											2013-05-30 00:07:32 -04:00
										 |  |  |  |                        _("<schema id='%s'> is list of not yet existing " | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                          "schema “%s”"), id, list_of); | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-09-09 16:28:18 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (tmp->path) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                        _("Cannot be a list of a schema with a path")); | 
					
						
							| 
									
										
										
										
											2010-09-09 16:28:18 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (extends) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-09-09 16:28:18 -04:00
										 |  |  |  |       if (extends->path) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                        _("Cannot extend a schema with a path")); | 
					
						
							| 
									
										
										
										
											2010-09-09 16:28:18 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |       if (list_of) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           if (extends->list_of == NULL) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                            _("<schema id='%s'> is a list, extending " | 
					
						
							|  |  |  |  |                              "<schema id='%s'> which is not a list"), | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |                            id, extends_name); | 
					
						
							|  |  |  |  |               return; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           if (!is_subclass (list_of, extends->list_of, state->schema_table)) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                            _("<schema id='%s' list-of='%s'> extends <schema " | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                              "id='%s' list-of='%s'> but “%s” does not " | 
					
						
							|  |  |  |  |                              "extend “%s”"), id, list_of, extends_name, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                            extends->list_of, list_of, extends->list_of); | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |               return; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         /* by default we are a list of the same thing that the schema
 | 
					
						
							|  |  |  |  |          * we are extending is a list of (which might be nothing) | 
					
						
							|  |  |  |  |          */ | 
					
						
							|  |  |  |  |         list_of = extends->list_of; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   if (path && !(g_str_has_prefix (path, "/") && g_str_has_suffix (path, "/"))) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |       g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                    _("A path, if given, must begin and end with a slash")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-09 16:28:18 -04:00
										 |  |  |  |   if (path && list_of && !g_str_has_suffix (path, ":/")) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                    _("The path of a list must end with “:/”")); | 
					
						
							| 
									
										
										
										
											2010-09-09 16:28:18 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-12 19:55:34 -04:00
										 |  |  |  |   if (path && (g_str_has_prefix (path, "/apps/") || | 
					
						
							|  |  |  |  |                g_str_has_prefix (path, "/desktop/") || | 
					
						
							|  |  |  |  |                g_str_has_prefix (path, "/system/"))) | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       gchar *message = NULL; | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |       message = g_strdup_printf (_("Warning: Schema “%s” has path “%s”.  " | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                                    "Paths starting with " | 
					
						
							| 
									
										
										
										
											2017-08-03 16:31:37 +01:00
										 |  |  |  |                                    "“/apps/”, “/desktop/” or “/system/” are deprecated."), | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                                  id, path); | 
					
						
							|  |  |  |  |       g_printerr ("%s\n", message); | 
					
						
							|  |  |  |  |       g_free (message); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-04-12 19:55:34 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |   state->schema_state = schema_state_new (path, gettext_domain, | 
					
						
							|  |  |  |  |                                           extends, extends_name, list_of); | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   my_id = g_strdup (id); | 
					
						
							|  |  |  |  |   state->this_file_schemas = g_slist_prepend (state->this_file_schemas, my_id); | 
					
						
							|  |  |  |  |   g_hash_table_insert (state->schema_table, my_id, state->schema_state); | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | parse_state_start_enum (ParseState   *state, | 
					
						
							|  |  |  |  |                         const gchar  *id, | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                         gboolean      is_flags, | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                         GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |   GSList **list = is_flags ? &state->this_file_flagss : &state->this_file_enums; | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   GHashTable *table = is_flags ? state->flags_table : state->enum_table; | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |   gchar *my_id; | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_hash_table_lookup (table, id)) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_MARKUP_ERROR, | 
					
						
							|  |  |  |  |                    G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                    _("<%s id='%s'> already specified"), | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                    is_flags ? "flags" : "enum", id); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |   state->enum_state = enum_state_new (is_flags); | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   my_id = g_strdup (id); | 
					
						
							|  |  |  |  |   *list = g_slist_prepend (*list, my_id); | 
					
						
							|  |  |  |  |   g_hash_table_insert (table, my_id, state->enum_state); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* GMarkup Parser Functions {{{1 */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Start element {{{2 */ | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | start_element (GMarkupParseContext  *context, | 
					
						
							|  |  |  |  |                const gchar          *element_name, | 
					
						
							|  |  |  |  |                const gchar         **attribute_names, | 
					
						
							|  |  |  |  |                const gchar         **attribute_values, | 
					
						
							|  |  |  |  |                gpointer              user_data, | 
					
						
							|  |  |  |  |                GError              **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   ParseState *state = user_data; | 
					
						
							|  |  |  |  |   const GSList *element_stack; | 
					
						
							|  |  |  |  |   const gchar *container; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   element_stack = g_markup_parse_context_get_element_stack (context); | 
					
						
							|  |  |  |  |   container = element_stack->next ? element_stack->next->data : NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #define COLLECT(first, ...) \
 | 
					
						
							|  |  |  |  |   g_markup_collect_attributes (element_name,                                 \ | 
					
						
							|  |  |  |  |                                attribute_names, attribute_values, error,     \ | 
					
						
							|  |  |  |  |                                first, __VA_ARGS__, G_MARKUP_COLLECT_INVALID) | 
					
						
							|  |  |  |  | #define OPTIONAL   G_MARKUP_COLLECT_OPTIONAL
 | 
					
						
							|  |  |  |  | #define STRDUP     G_MARKUP_COLLECT_STRDUP
 | 
					
						
							|  |  |  |  | #define STRING     G_MARKUP_COLLECT_STRING
 | 
					
						
							| 
									
										
										
										
											2010-04-21 17:44:47 -04:00
										 |  |  |  | #define NO_ATTRS()  COLLECT (G_MARKUP_COLLECT_INVALID, NULL)
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   /* Toplevel items {{{3 */ | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   if (container == NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (element_name, "schemalist") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           COLLECT (OPTIONAL | STRDUP, | 
					
						
							|  |  |  |  |                    "gettext-domain", | 
					
						
							|  |  |  |  |                    &state->schemalist_domain); | 
					
						
							|  |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* children of <schemalist> {{{3 */ | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   else if (strcmp (container, "schemalist") == 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (element_name, "schema") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |           const gchar *id, *path, *gettext_domain, *extends, *list_of; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           if (COLLECT (STRING, "id", &id, | 
					
						
							|  |  |  |  |                        OPTIONAL | STRING, "path", &path, | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |                        OPTIONAL | STRING, "gettext-domain", &gettext_domain, | 
					
						
							|  |  |  |  |                        OPTIONAL | STRING, "extends", &extends, | 
					
						
							|  |  |  |  |                        OPTIONAL | STRING, "list-of", &list_of)) | 
					
						
							| 
									
										
										
										
											2010-11-26 21:05:20 +01:00
										 |  |  |  |             parse_state_start_schema (state, id, path, | 
					
						
							|  |  |  |  |                                       gettext_domain ? gettext_domain | 
					
						
							|  |  |  |  |                                                      : state->schemalist_domain, | 
					
						
							| 
									
										
										
										
											2010-06-29 14:41:04 -04:00
										 |  |  |  |                                       extends, list_of, error); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |       else if (strcmp (element_name, "enum") == 0) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |         { | 
					
						
							|  |  |  |  |           const gchar *id; | 
					
						
							|  |  |  |  |           if (COLLECT (STRING, "id", &id)) | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |             parse_state_start_enum (state, id, FALSE, error); | 
					
						
							|  |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       else if (strcmp (element_name, "flags") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           const gchar *id; | 
					
						
							|  |  |  |  |           if (COLLECT (STRING, "id", &id)) | 
					
						
							|  |  |  |  |             parse_state_start_enum (state, id, TRUE, error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* children of <schema> {{{3 */ | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   else if (strcmp (container, "schema") == 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (element_name, "key") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |           const gchar *name, *type_string, *enum_type, *flags_type; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |           if (COLLECT (STRING,            "name",  &name, | 
					
						
							|  |  |  |  |                        OPTIONAL | STRING, "type",  &type_string, | 
					
						
							|  |  |  |  |                        OPTIONAL | STRING, "enum",  &enum_type, | 
					
						
							|  |  |  |  |                        OPTIONAL | STRING, "flags", &flags_type)) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |             state->key_state = schema_state_add_key (state->schema_state, | 
					
						
							|  |  |  |  |                                                      state->enum_table, | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                                                      state->flags_table, | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                                                      name, type_string, | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  |                                                      enum_type, flags_type, | 
					
						
							|  |  |  |  |                                                      error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       else if (strcmp (element_name, "child") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           const gchar *name, *schema; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           if (COLLECT (STRING, "name", &name, STRING, "schema", &schema)) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |             schema_state_add_child (state->schema_state, | 
					
						
							|  |  |  |  |                                     name, schema, error); | 
					
						
							|  |  |  |  |           return; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |       else if (strcmp (element_name, "override") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-01-19 18:45:20 +01:00
										 |  |  |  |           const gchar *name, *l10n, *str_context; | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-19 18:45:20 +01:00
										 |  |  |  |           if (COLLECT (STRING, "name", &name, | 
					
						
							|  |  |  |  |                        OPTIONAL | STRING, "l10n", &l10n, | 
					
						
							|  |  |  |  |                        OPTIONAL | STRING, "context", &str_context)) | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |             schema_state_add_override (state->schema_state, | 
					
						
							|  |  |  |  |                                        &state->key_state, &state->string, | 
					
						
							| 
									
										
										
										
											2022-01-19 18:45:20 +01:00
										 |  |  |  |                                        name, l10n, str_context, error); | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* children of <key> {{{3 */ | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   else if (strcmp (container, "key") == 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (element_name, "default") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-01-19 18:45:20 +01:00
										 |  |  |  |           const gchar *l10n, *str_context; | 
					
						
							|  |  |  |  |           if (COLLECT (STRING | OPTIONAL, "l10n", &l10n, | 
					
						
							|  |  |  |  |                        STRING | OPTIONAL, "context", &str_context)) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |             state->string = key_state_start_default (state->key_state, | 
					
						
							| 
									
										
										
										
											2022-01-19 18:45:20 +01:00
										 |  |  |  |                                                      l10n, str_context, error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-01 18:55:54 -04:00
										 |  |  |  |       else if (strcmp (element_name, "summary") == 0) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2015-04-01 18:55:54 -04:00
										 |  |  |  |           if (NO_ATTRS ()) | 
					
						
							| 
									
										
										
										
											2015-04-08 22:08:13 -04:00
										 |  |  |  |             { | 
					
						
							|  |  |  |  |               if (state->key_state->summary_seen && state->strict) | 
					
						
							|  |  |  |  |                 g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							|  |  |  |  |                              _("Only one <%s> element allowed inside <%s>"), | 
					
						
							|  |  |  |  |                              element_name, container); | 
					
						
							|  |  |  |  |               else | 
					
						
							|  |  |  |  |                 state->string = g_string_new (NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               state->key_state->summary_seen = TRUE; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-04-01 18:55:54 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       else if (strcmp (element_name, "description") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           if (NO_ATTRS ()) | 
					
						
							| 
									
										
										
										
											2015-04-08 22:08:13 -04:00
										 |  |  |  |             { | 
					
						
							|  |  |  |  |               if (state->key_state->description_seen && state->strict) | 
					
						
							|  |  |  |  |                 g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							|  |  |  |  |                              _("Only one <%s> element allowed inside <%s>"), | 
					
						
							|  |  |  |  |                              element_name, container); | 
					
						
							|  |  |  |  |               else | 
					
						
							|  |  |  |  |                 state->string = g_string_new (NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             state->key_state->description_seen = TRUE; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |       else if (strcmp (element_name, "range") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           const gchar *min, *max; | 
					
						
							| 
									
										
										
										
											2011-03-31 15:10:25 +05:30
										 |  |  |  |           if (COLLECT (STRING | OPTIONAL, "min", &min, | 
					
						
							|  |  |  |  |                        STRING | OPTIONAL, "max", &max)) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |             key_state_set_range (state->key_state, min, max, error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  |       else if (strcmp (element_name, "choices") == 0) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           if (NO_ATTRS ()) | 
					
						
							|  |  |  |  |             key_state_start_choices (state->key_state, error); | 
					
						
							|  |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       else if (strcmp (element_name, "aliases") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           if (NO_ATTRS ()) | 
					
						
							|  |  |  |  |             key_state_start_aliases (state->key_state, error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* children of <choices> {{{3 */ | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  |   else if (strcmp (container, "choices") == 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (element_name, "choice") == 0) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  |           const gchar *value; | 
					
						
							|  |  |  |  |           if (COLLECT (STRING, "value", &value)) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |             key_state_add_choice (state->key_state, value, error); | 
					
						
							|  |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   /* children of <aliases> {{{3 */ | 
					
						
							|  |  |  |  |   else if (strcmp (container, "aliases") == 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (element_name, "alias") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           const gchar *value, *target; | 
					
						
							|  |  |  |  |           if (COLLECT (STRING, "value", &value, STRING, "target", &target)) | 
					
						
							|  |  |  |  |             key_state_add_alias (state->key_state, value, target, error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* children of <enum> {{{3 */ | 
					
						
							| 
									
										
										
										
											2010-06-30 13:27:38 -04:00
										 |  |  |  |   else if (strcmp (container, "enum") == 0 || | 
					
						
							|  |  |  |  |            strcmp (container, "flags") == 0) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (element_name, "value") == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           const gchar *nick, *valuestr; | 
					
						
							|  |  |  |  |           if (COLLECT (STRING, "nick", &nick, | 
					
						
							|  |  |  |  |                        STRING, "value", &valuestr)) | 
					
						
							|  |  |  |  |             enum_state_add_value (state->enum_state, nick, valuestr, error); | 
					
						
							|  |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   /* 3}}} */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   if (container) | 
					
						
							|  |  |  |  |     g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_UNKNOWN_ELEMENT, | 
					
						
							| 
									
										
										
										
											2010-07-30 21:10:53 +02:00
										 |  |  |  |                  _("Element <%s> not allowed inside <%s>"), | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |                  element_name, container); | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_UNKNOWN_ELEMENT, | 
					
						
							| 
									
										
										
										
											2012-08-16 23:02:41 -04:00
										 |  |  |  |                  _("Element <%s> not allowed at the top level"), element_name); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | /* 2}}} */ | 
					
						
							|  |  |  |  | /* End element {{{2 */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | key_state_end (KeyState **state_ptr, | 
					
						
							|  |  |  |  |                GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   KeyState *state; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   state = *state_ptr; | 
					
						
							|  |  |  |  |   *state_ptr = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->default_value == NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error_literal (error, | 
					
						
							|  |  |  |  |                            G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                            _("Element <default> is required in <key>")); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | schema_state_end (SchemaState **state_ptr, | 
					
						
							|  |  |  |  |                   GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   *state_ptr = NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | end_element (GMarkupParseContext  *context, | 
					
						
							|  |  |  |  |              const gchar          *element_name, | 
					
						
							|  |  |  |  |              gpointer              user_data, | 
					
						
							|  |  |  |  |              GError              **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   ParseState *state = user_data; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   if (strcmp (element_name, "schemalist") == 0) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |       g_free (state->schemalist_domain); | 
					
						
							|  |  |  |  |       state->schemalist_domain = NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-30 13:27:38 -04:00
										 |  |  |  |   else if (strcmp (element_name, "enum") == 0 || | 
					
						
							|  |  |  |  |            strcmp (element_name, "flags") == 0) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     enum_state_end (&state->enum_state, error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   else if (strcmp (element_name, "schema") == 0) | 
					
						
							|  |  |  |  |     schema_state_end (&state->schema_state, error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |   else if (strcmp (element_name, "override") == 0) | 
					
						
							|  |  |  |  |     override_state_end (&state->key_state, &state->string, error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   else if (strcmp (element_name, "key") == 0) | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |     key_state_end (&state->key_state, error); | 
					
						
							| 
									
										
										
										
											2010-04-19 11:31:21 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   else if (strcmp (element_name, "default") == 0) | 
					
						
							|  |  |  |  |     key_state_end_default (state->key_state, &state->string, error); | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   else if (strcmp (element_name, "choices") == 0) | 
					
						
							|  |  |  |  |     key_state_end_choices (state->key_state, error); | 
					
						
							| 
									
										
										
										
											2010-04-25 19:59:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   else if (strcmp (element_name, "aliases") == 0) | 
					
						
							|  |  |  |  |     key_state_end_aliases (state->key_state, error); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   if (state->string) | 
					
						
							| 
									
										
										
										
											2010-04-21 18:08:05 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       g_string_free (state->string, TRUE); | 
					
						
							|  |  |  |  |       state->string = NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | /* Text {{{2 */ | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | text (GMarkupParseContext  *context, | 
					
						
							|  |  |  |  |       const gchar          *text, | 
					
						
							|  |  |  |  |       gsize                 text_len, | 
					
						
							|  |  |  |  |       gpointer              user_data, | 
					
						
							|  |  |  |  |       GError              **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   ParseState *state = user_data; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-28 01:09:08 +01:00
										 |  |  |  |   if (state->string) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       /* we are expecting a string, so store the text data.
 | 
					
						
							|  |  |  |  |        * | 
					
						
							|  |  |  |  |        * we store the data verbatim here and deal with whitespace | 
					
						
							|  |  |  |  |        * later on.  there are two reasons for that: | 
					
						
							|  |  |  |  |        * | 
					
						
							|  |  |  |  |        *  1) whitespace is handled differently depending on the tag | 
					
						
							|  |  |  |  |        *     type. | 
					
						
							|  |  |  |  |        * | 
					
						
							|  |  |  |  |        *  2) we could do leading whitespace removal by refusing to | 
					
						
							|  |  |  |  |        *     insert it into state->string if it's at the start, but for | 
					
						
							|  |  |  |  |        *     trailing whitespace, we have no idea if there is another | 
					
						
							|  |  |  |  |        *     text() call coming or not. | 
					
						
							|  |  |  |  |        */ | 
					
						
							|  |  |  |  |       g_string_append_len (state->string, text, text_len); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       /* string is not expected: accept (and ignore) pure whitespace */ | 
					
						
							|  |  |  |  |       gsize i; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-28 01:09:08 +01:00
										 |  |  |  |       for (i = 0; i < text_len; i++) | 
					
						
							|  |  |  |  |         if (!g_ascii_isspace (text[i])) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |                          _("Text may not appear inside <%s>"), | 
					
						
							| 
									
										
										
										
											2012-01-28 01:09:08 +01:00
										 |  |  |  |                          g_markup_parse_context_get_element (context)); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | /* Write to GVDB {{{1 */ | 
					
						
							|  |  |  |  | typedef struct | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GHashTable *table; | 
					
						
							|  |  |  |  |   GvdbItem *root; | 
					
						
							|  |  |  |  | } GvdbPair; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | gvdb_pair_init (GvdbPair *pair) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   pair->table = gvdb_hash_table_new (NULL, NULL); | 
					
						
							|  |  |  |  |   pair->root = gvdb_hash_table_insert (pair->table, ""); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | gvdb_pair_clear (GvdbPair *pair) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_hash_table_unref (pair->table); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | typedef struct | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   GHashTable *schema_table; | 
					
						
							|  |  |  |  |   GvdbPair root_pair; | 
					
						
							|  |  |  |  | } WriteToFileData; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | typedef struct | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GHashTable *schema_table; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   GvdbPair pair; | 
					
						
							|  |  |  |  |   gboolean l10n; | 
					
						
							|  |  |  |  | } OutputSchemaData; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | output_key (gpointer key, | 
					
						
							|  |  |  |  |             gpointer value, | 
					
						
							|  |  |  |  |             gpointer user_data) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   OutputSchemaData *data; | 
					
						
							|  |  |  |  |   const gchar *name; | 
					
						
							|  |  |  |  |   KeyState *state; | 
					
						
							|  |  |  |  |   GvdbItem *item; | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   GVariant *serialised = NULL; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   name = key; | 
					
						
							|  |  |  |  |   state = value; | 
					
						
							|  |  |  |  |   data = user_data; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   item = gvdb_hash_table_insert (data->pair.table, name); | 
					
						
							|  |  |  |  |   gvdb_item_set_parent (item, data->pair.root); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   serialised = key_state_serialise (state); | 
					
						
							|  |  |  |  |   gvdb_item_set_value (item, serialised); | 
					
						
							|  |  |  |  |   g_variant_unref (serialised); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->l10n) | 
					
						
							|  |  |  |  |     data->l10n = TRUE; | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->child_schema && | 
					
						
							|  |  |  |  |       !g_hash_table_lookup (data->schema_table, state->child_schema)) | 
					
						
							| 
									
										
										
										
											2017-08-03 11:34:08 +01:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       gchar *message = NULL; | 
					
						
							|  |  |  |  |       message = g_strdup_printf (_("Warning: undefined reference to <schema id='%s'/>"), | 
					
						
							|  |  |  |  |                                  state->child_schema); | 
					
						
							|  |  |  |  |       g_printerr ("%s\n", message); | 
					
						
							|  |  |  |  |       g_free (message); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | output_schema (gpointer key, | 
					
						
							|  |  |  |  |                gpointer value, | 
					
						
							|  |  |  |  |                gpointer user_data) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   WriteToFileData *wtf_data = user_data; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   OutputSchemaData data; | 
					
						
							|  |  |  |  |   GvdbPair *root_pair; | 
					
						
							|  |  |  |  |   SchemaState *state; | 
					
						
							|  |  |  |  |   const gchar *id; | 
					
						
							|  |  |  |  |   GvdbItem *item; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   id = key; | 
					
						
							|  |  |  |  |   state = value; | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   root_pair = &wtf_data->root_pair; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   data.schema_table = wtf_data->schema_table; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   gvdb_pair_init (&data.pair); | 
					
						
							|  |  |  |  |   data.l10n = FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   item = gvdb_hash_table_insert (root_pair->table, id); | 
					
						
							|  |  |  |  |   gvdb_item_set_parent (item, root_pair->root); | 
					
						
							|  |  |  |  |   gvdb_item_set_hash_table (item, data.pair.table); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_hash_table_foreach (state->keys, output_key, &data); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->path) | 
					
						
							|  |  |  |  |     gvdb_hash_table_insert_string (data.pair.table, ".path", state->path); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  |   if (state->extends_name) | 
					
						
							|  |  |  |  |     gvdb_hash_table_insert_string (data.pair.table, ".extends", | 
					
						
							|  |  |  |  |                                    state->extends_name); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (state->list_of) | 
					
						
							|  |  |  |  |     gvdb_hash_table_insert_string (data.pair.table, ".list-of", | 
					
						
							| 
									
										
										
										
											2011-01-30 20:45:52 +01:00
										 |  |  |  |                                    state->list_of); | 
					
						
							| 
									
										
										
										
											2010-06-29 20:24:39 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   if (data.l10n) | 
					
						
							|  |  |  |  |     gvdb_hash_table_insert_string (data.pair.table, | 
					
						
							|  |  |  |  |                                    ".gettext-domain", | 
					
						
							|  |  |  |  |                                    state->gettext_domain); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   gvdb_pair_clear (&data.pair); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							|  |  |  |  | write_to_file (GHashTable   *schema_table, | 
					
						
							|  |  |  |  |                const gchar  *filename, | 
					
						
							|  |  |  |  |                GError      **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   WriteToFileData data; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   gboolean success; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   data.schema_table = schema_table; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   gvdb_pair_init (&data.root_pair); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   g_hash_table_foreach (schema_table, output_schema, &data); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   success = gvdb_table_write_contents (data.root_pair.table, filename, | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |                                        G_BYTE_ORDER != G_LITTLE_ENDIAN, | 
					
						
							|  |  |  |  |                                        error); | 
					
						
							| 
									
										
										
										
											2011-04-08 01:47:52 -04:00
										 |  |  |  |   g_hash_table_unref (data.root_pair.table); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   return success; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Parser driver {{{1 */ | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | static GHashTable * | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  | parse_gschema_files (gchar    **files, | 
					
						
							|  |  |  |  |                      gboolean   strict) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-17 22:18:16 +01:00
										 |  |  |  |   GMarkupParser parser = { start_element, end_element, text, NULL, NULL }; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   ParseState state = { 0, }; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   const gchar *filename; | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												gsettings: stay compatible with installed schemas
Bug 747209 introduced an error when multiple <summary> or <description>
tags are found for a single key in a GSettings schema.  This check
should have been present from the start, but it was left out because the
schema compiler doesn't include these items in the cache file.  Even
still -- part of the schema compiler's job is validation, and it should
be enforcing proper syntax here.
Repeated <summary> and <description> tags are a semi-common problem when
intltool has been misconfigured in the build system of a package, but
it's possible to imagine mistakes being made by hand as well.
The idea is that these problems would be caught during the build of a
package and maintainers would be forced to fix their build systems.
An unintended side-effect of this change, however, is that the schema
compiler started ignoring already-installed schemas that contained these
problems, when rebuilding the cache.  This means that the installation
of _any_ application would cause the regeneration of the entire cache,
with these already-installed applications being excluded.  Without the
schema in the cache, the application would crash on next startup.
The validation check in the gsettings m4 macro passes --strict to the
compiler, which is not used when rebuilding the cache after
installation.  Pass this flag down into the parser and only throw the
error in case --strict was given.  This will result in the (desired)
build failure without also causing already-installed apps to stop
functioning.
This means that we will not get even a warning about the invalid schema
file in the already-installed case, but that's fine.  There is no sense
spamming the user with these messages when they are already quite fatal
for the developer at build time.
https://bugzilla.gnome.org/show_bug.cgi?id=747472
											
										 
											2015-04-08 21:55:58 -04:00
										 |  |  |  |   state.strict = strict; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   state.enum_table = g_hash_table_new_full (g_str_hash, g_str_equal, | 
					
						
							|  |  |  |  |                                             g_free, enum_state_free); | 
					
						
							| 
									
										
										
										
											2010-07-01 18:58:56 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   state.flags_table = g_hash_table_new_full (g_str_hash, g_str_equal, | 
					
						
							|  |  |  |  |                                              g_free, enum_state_free); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   state.schema_table = g_hash_table_new_full (g_str_hash, g_str_equal, | 
					
						
							|  |  |  |  |                                               g_free, schema_state_free); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   while ((filename = *files++) != NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-07-02 11:14:28 -04:00
										 |  |  |  |       GMarkupParseContext *context; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |       gchar *contents; | 
					
						
							|  |  |  |  |       gsize size; | 
					
						
							| 
									
										
										
										
											2015-05-15 22:41:29 -04:00
										 |  |  |  |       gint line, col; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |       if (!g_file_get_contents (filename, &contents, &size, &error)) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           fprintf (stderr, "%s\n", error->message); | 
					
						
							|  |  |  |  |           g_clear_error (&error); | 
					
						
							|  |  |  |  |           continue; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-02 11:14:28 -04:00
										 |  |  |  |       context = g_markup_parse_context_new (&parser, | 
					
						
							| 
									
										
										
										
											2011-11-07 00:13:56 -05:00
										 |  |  |  |                                             G_MARKUP_TREAT_CDATA_AS_TEXT | | 
					
						
							| 
									
										
										
										
											2013-10-28 15:30:25 -07:00
										 |  |  |  |                                             G_MARKUP_PREFIX_ERROR_POSITION | | 
					
						
							|  |  |  |  |                                             G_MARKUP_IGNORE_QUALIFIED, | 
					
						
							| 
									
										
										
										
											2010-07-02 11:14:28 -04:00
										 |  |  |  |                                             &state, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |       if (!g_markup_parse_context_parse (context, contents, size, &error) || | 
					
						
							|  |  |  |  |           !g_markup_parse_context_end_parse (context, &error)) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |           GSList *item; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |           /* back out any changes from this file */ | 
					
						
							|  |  |  |  |           for (item = state.this_file_schemas; item; item = item->next) | 
					
						
							|  |  |  |  |             g_hash_table_remove (state.schema_table, item->data); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           for (item = state.this_file_flagss; item; item = item->next) | 
					
						
							|  |  |  |  |             g_hash_table_remove (state.flags_table, item->data); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           for (item = state.this_file_enums; item; item = item->next) | 
					
						
							|  |  |  |  |             g_hash_table_remove (state.enum_table, item->data); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           /* let them know */ | 
					
						
							| 
									
										
										
										
											2015-05-15 22:41:29 -04:00
										 |  |  |  |           g_markup_parse_context_get_position (context, &line, &col); | 
					
						
							|  |  |  |  |           fprintf (stderr, "%s:%d:%d  %s.  ", filename, line, col, error->message); | 
					
						
							| 
									
										
										
										
											2010-10-16 23:30:30 -04:00
										 |  |  |  |           g_clear_error (&error); | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |           if (strict) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               /* Translators: Do not translate "--strict". */ | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |               fprintf (stderr, "%s\n", _("--strict was specified; exiting.")); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |               g_hash_table_unref (state.schema_table); | 
					
						
							|  |  |  |  |               g_hash_table_unref (state.flags_table); | 
					
						
							|  |  |  |  |               g_hash_table_unref (state.enum_table); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |               g_free (contents); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |               return NULL; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |           else | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |             { | 
					
						
							|  |  |  |  |               fprintf (stderr, "%s\n", _("This entire file has been ignored.")); | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-07-02 11:14:28 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |       /* cleanup */ | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |       g_free (contents); | 
					
						
							| 
									
										
										
										
											2010-07-02 11:14:28 -04:00
										 |  |  |  |       g_markup_parse_context_free (context); | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |       g_slist_free (state.this_file_schemas); | 
					
						
							|  |  |  |  |       g_slist_free (state.this_file_flagss); | 
					
						
							|  |  |  |  |       g_slist_free (state.this_file_enums); | 
					
						
							|  |  |  |  |       state.this_file_schemas = NULL; | 
					
						
							|  |  |  |  |       state.this_file_flagss = NULL; | 
					
						
							|  |  |  |  |       state.this_file_enums = NULL; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |   g_hash_table_unref (state.flags_table); | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |   g_hash_table_unref (state.enum_table); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return state.schema_table; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gint | 
					
						
							|  |  |  |  | compare_strings (gconstpointer a, | 
					
						
							|  |  |  |  |                  gconstpointer b) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-12-20 00:22:05 +01:00
										 |  |  |  |   const gchar *one = a; | 
					
						
							| 
									
										
										
										
											2023-01-09 12:20:10 +00:00
										 |  |  |  |   const gchar *two = b; | 
					
						
							| 
									
										
										
										
											2010-07-02 11:11:45 -04:00
										 |  |  |  |   gint cmp; | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-02 11:11:45 -04:00
										 |  |  |  |   cmp = g_str_has_suffix (two, ".enums.xml") - | 
					
						
							|  |  |  |  |         g_str_has_suffix (one, ".enums.xml"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!cmp) | 
					
						
							|  |  |  |  |     cmp = strcmp (one, two); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return cmp; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  | static gboolean | 
					
						
							|  |  |  |  | set_overrides (GHashTable  *schema_table, | 
					
						
							|  |  |  |  |                gchar      **files, | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                gboolean     strict) | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   const gchar *filename; | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   while ((filename = *files++)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       GKeyFile *key_file; | 
					
						
							|  |  |  |  |       gchar **groups; | 
					
						
							|  |  |  |  |       gint i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |       g_debug ("Processing override file '%s'", filename); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |       key_file = g_key_file_new (); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |       if (!g_key_file_load_from_file (key_file, filename, 0, &error)) | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |           fprintf (stderr, "%s: %s.  ", filename, error->message); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |           g_key_file_free (key_file); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |           g_clear_error (&error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           if (!strict) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |               fprintf (stderr, "%s\n", _("Ignoring this file.")); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |               continue; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |           fprintf (stderr, "%s\n", _("--strict was specified; exiting.")); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |           return FALSE; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       groups = g_key_file_get_groups (key_file, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       for (i = 0; groups[i]; i++) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           const gchar *group = groups[i]; | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |           const gchar *schema_name; | 
					
						
							|  |  |  |  |           const gchar *desktop_id; | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |           SchemaState *schema; | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |           gchar **pieces; | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |           gchar **keys; | 
					
						
							|  |  |  |  |           gint j; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |           pieces = g_strsplit (group, ":", 2); | 
					
						
							|  |  |  |  |           schema_name = pieces[0]; | 
					
						
							|  |  |  |  |           desktop_id = pieces[1]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           g_debug ("Processing group '%s' (schema '%s', %s)", | 
					
						
							|  |  |  |  |                    group, schema_name, desktop_id ? desktop_id : "all desktops"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           schema = g_hash_table_lookup (schema_table, schema_name); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |           if (schema == NULL) | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |             { | 
					
						
							|  |  |  |  |               /* Having the schema not be installed is expected to be a
 | 
					
						
							|  |  |  |  |                * common case.  Don't even emit an error message about | 
					
						
							|  |  |  |  |                * that. | 
					
						
							|  |  |  |  |                */ | 
					
						
							|  |  |  |  |               g_strfreev (pieces); | 
					
						
							|  |  |  |  |               continue; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |           keys = g_key_file_get_keys (key_file, group, NULL, NULL); | 
					
						
							|  |  |  |  |           g_assert (keys != NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           for (j = 0; keys[j]; j++) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               const gchar *key = keys[j]; | 
					
						
							|  |  |  |  |               KeyState *state; | 
					
						
							|  |  |  |  |               GVariant *value; | 
					
						
							|  |  |  |  |               gchar *string; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               state = g_hash_table_lookup (schema->keys, key); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               if (state == NULL) | 
					
						
							|  |  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                   if (!strict) | 
					
						
							|  |  |  |  |                     { | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                       fprintf (stderr, _("No such key “%s” in schema “%s” as " | 
					
						
							|  |  |  |  |                                          "specified in override file “%s”; " | 
					
						
							|  |  |  |  |                                          "ignoring override for this key."), | 
					
						
							|  |  |  |  |                                key, group, filename); | 
					
						
							|  |  |  |  |                       fprintf (stderr, "\n"); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                       continue; | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                   fprintf (stderr, _("No such key “%s” in schema “%s” as " | 
					
						
							|  |  |  |  |                                      "specified in override file “%s” and " | 
					
						
							|  |  |  |  |                                      "--strict was specified; exiting."), | 
					
						
							|  |  |  |  |                            key, group, filename); | 
					
						
							|  |  |  |  |                   fprintf (stderr, "\n"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |                   g_key_file_free (key_file); | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |                   g_strfreev (pieces); | 
					
						
							|  |  |  |  |                   g_strfreev (groups); | 
					
						
							|  |  |  |  |                   g_strfreev (keys); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                   return FALSE; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               if (desktop_id != NULL && state->l10n) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                   /* Let's avoid the n*m case of per-desktop localised
 | 
					
						
							|  |  |  |  |                    * default values, and just forbid it. | 
					
						
							|  |  |  |  |                    */ | 
					
						
							|  |  |  |  |                   if (!strict) | 
					
						
							|  |  |  |  |                     { | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                       fprintf (stderr, | 
					
						
							|  |  |  |  |                                _("Cannot provide per-desktop overrides for " | 
					
						
							| 
									
										
										
										
											2019-08-14 10:38:23 +02:00
										 |  |  |  |                                  "localized key “%s” in schema “%s” (override " | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                                  "file “%s”); ignoring override for this key."), | 
					
						
							|  |  |  |  |                            key, group, filename); | 
					
						
							|  |  |  |  |                       fprintf (stderr, "\n"); | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |                       continue; | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                   fprintf (stderr, | 
					
						
							|  |  |  |  |                            _("Cannot provide per-desktop overrides for " | 
					
						
							| 
									
										
										
										
											2019-08-14 10:38:23 +02:00
										 |  |  |  |                              "localized key “%s” in schema “%s” (override " | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                              "file “%s”) and --strict was specified; exiting."), | 
					
						
							|  |  |  |  |                            key, group, filename); | 
					
						
							|  |  |  |  |                   fprintf (stderr, "\n"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |                   g_key_file_free (key_file); | 
					
						
							|  |  |  |  |                   g_strfreev (pieces); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |                   g_strfreev (groups); | 
					
						
							|  |  |  |  |                   g_strfreev (keys); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                   return FALSE; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               string = g_key_file_get_value (key_file, group, key, NULL); | 
					
						
							|  |  |  |  |               g_assert (string != NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               value = g_variant_parse (state->type, string, | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                                        NULL, NULL, &error); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |               if (value == NULL) | 
					
						
							|  |  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                   if (!strict) | 
					
						
							|  |  |  |  |                     { | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                       fprintf (stderr, _("Error parsing key “%s” in schema “%s” " | 
					
						
							|  |  |  |  |                                          "as specified in override file “%s”: " | 
					
						
							|  |  |  |  |                                          "%s. Ignoring override for this key."), | 
					
						
							|  |  |  |  |                                key, group, filename, error->message); | 
					
						
							|  |  |  |  |                       fprintf (stderr, "\n"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                       g_clear_error (&error); | 
					
						
							|  |  |  |  |                       g_free (string); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                       continue; | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                   fprintf (stderr, _("Error parsing key “%s” in schema “%s” " | 
					
						
							|  |  |  |  |                                      "as specified in override file “%s”: " | 
					
						
							|  |  |  |  |                                      "%s. --strict was specified; exiting."), | 
					
						
							|  |  |  |  |                            key, group, filename, error->message); | 
					
						
							|  |  |  |  |                   fprintf (stderr, "\n"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                   g_clear_error (&error); | 
					
						
							|  |  |  |  |                   g_free (string); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |                   g_key_file_free (key_file); | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |                   g_strfreev (pieces); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |                   g_strfreev (groups); | 
					
						
							|  |  |  |  |                   g_strfreev (keys); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                   return FALSE; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               if (state->minimum) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                   if (g_variant_compare (value, state->minimum) < 0 || | 
					
						
							|  |  |  |  |                       g_variant_compare (value, state->maximum) > 0) | 
					
						
							|  |  |  |  |                     { | 
					
						
							|  |  |  |  |                       g_variant_unref (value); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                       g_free (string); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                       if (!strict) | 
					
						
							|  |  |  |  |                         { | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                           fprintf (stderr, | 
					
						
							|  |  |  |  |                                    _("Override for key “%s” in schema “%s” in " | 
					
						
							|  |  |  |  |                                      "override file “%s” is outside the range " | 
					
						
							|  |  |  |  |                                      "given in the schema; ignoring override " | 
					
						
							|  |  |  |  |                                      "for this key."), | 
					
						
							|  |  |  |  |                                    key, group, filename); | 
					
						
							|  |  |  |  |                           fprintf (stderr, "\n"); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                           continue; | 
					
						
							|  |  |  |  |                         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                       fprintf (stderr, | 
					
						
							|  |  |  |  |                                _("Override for key “%s” in schema “%s” in " | 
					
						
							|  |  |  |  |                                  "override file “%s” is outside the range " | 
					
						
							|  |  |  |  |                                  "given in the schema and --strict was " | 
					
						
							|  |  |  |  |                                  "specified; exiting."), | 
					
						
							|  |  |  |  |                                key, group, filename); | 
					
						
							|  |  |  |  |                       fprintf (stderr, "\n"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                       g_key_file_free (key_file); | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |                       g_strfreev (pieces); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |                       g_strfreev (groups); | 
					
						
							|  |  |  |  |                       g_strfreev (keys); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                       return FALSE; | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               else if (state->strinfo->len) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                   if (!is_valid_choices (value, state->strinfo)) | 
					
						
							|  |  |  |  |                     { | 
					
						
							|  |  |  |  |                       g_variant_unref (value); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                       g_free (string); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                       if (!strict) | 
					
						
							|  |  |  |  |                         { | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                           fprintf (stderr, | 
					
						
							|  |  |  |  |                                    _("Override for key “%s” in schema “%s” in " | 
					
						
							|  |  |  |  |                                      "override file “%s” is not in the list " | 
					
						
							|  |  |  |  |                                      "of valid choices; ignoring override for " | 
					
						
							|  |  |  |  |                                      "this key."), | 
					
						
							|  |  |  |  |                                    key, group, filename); | 
					
						
							|  |  |  |  |                           fprintf (stderr, "\n"); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                           continue; | 
					
						
							|  |  |  |  |                         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |                       fprintf (stderr, | 
					
						
							|  |  |  |  |                                _("Override for key “%s” in schema “%s” in " | 
					
						
							|  |  |  |  |                                  "override file “%s” is not in the list " | 
					
						
							|  |  |  |  |                                  "of valid choices and --strict was specified; " | 
					
						
							|  |  |  |  |                                  "exiting."), | 
					
						
							|  |  |  |  |                                key, group, filename); | 
					
						
							|  |  |  |  |                       fprintf (stderr, "\n"); | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |                       g_key_file_free (key_file); | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |                       g_strfreev (pieces); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |                       g_strfreev (groups); | 
					
						
							|  |  |  |  |                       g_strfreev (keys); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                       return FALSE; | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |               if (desktop_id != NULL) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                   if (state->desktop_overrides == NULL) | 
					
						
							|  |  |  |  |                     state->desktop_overrides = g_variant_dict_new (NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                   g_variant_dict_insert_value (state->desktop_overrides, desktop_id, value); | 
					
						
							|  |  |  |  |                   g_variant_unref (value); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |               else | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                   g_variant_unref (state->default_value); | 
					
						
							|  |  |  |  |                   state->default_value = value; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |               g_free (string); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 11:10:18 +01:00
										 |  |  |  |           g_strfreev (pieces); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |           g_strfreev (keys); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       g_strfreev (groups); | 
					
						
							| 
									
										
										
										
											2018-08-01 13:01:51 +01:00
										 |  |  |  |       g_key_file_free (key_file); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | int | 
					
						
							|  |  |  |  | main (int argc, char **argv) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							|  |  |  |  |   GHashTable *table = NULL; | 
					
						
							|  |  |  |  |   GDir *dir = NULL; | 
					
						
							| 
									
										
										
										
											2010-04-19 15:48:30 +03:00
										 |  |  |  |   const gchar *file; | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   const gchar *srcdir; | 
					
						
							| 
									
										
										
										
											2016-09-30 22:56:21 +01:00
										 |  |  |  |   gboolean show_version_and_exit = FALSE; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   gchar *targetdir = NULL; | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   gchar *target = NULL; | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  |   gboolean dry_run = FALSE; | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |   gboolean strict = FALSE; | 
					
						
							| 
									
										
										
										
											2010-04-21 12:14:40 +01:00
										 |  |  |  |   gchar **schema_files = NULL; | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |   gchar **override_files = NULL; | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   GOptionContext *context = NULL; | 
					
						
							|  |  |  |  |   gint retval; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   GOptionEntry entries[] = { | 
					
						
							| 
									
										
										
										
											2016-09-30 22:56:21 +01:00
										 |  |  |  |     { "version", 0, 0, G_OPTION_ARG_NONE, &show_version_and_exit, N_("Show program version and exit"), NULL }, | 
					
						
							| 
									
										
										
										
											2019-02-18 17:21:13 +00:00
										 |  |  |  |     { "targetdir", 0, 0, G_OPTION_ARG_FILENAME, &targetdir, N_("Where to store the gschemas.compiled file"), N_("DIRECTORY") }, | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |     { "strict", 0, 0, G_OPTION_ARG_NONE, &strict, N_("Abort on any errors in schemas"), NULL }, | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  |     { "dry-run", 0, 0, G_OPTION_ARG_NONE, &dry_run, N_("Do not write the gschema.compiled file"), NULL }, | 
					
						
							| 
									
										
										
										
											2020-11-17 22:18:16 +01:00
										 |  |  |  |     { "allow-any-name", 0, 0, G_OPTION_ARG_NONE, &allow_any_name, N_("Do not enforce key name restrictions"), NULL }, | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* These options are only for use in the gschema-compile tests */ | 
					
						
							| 
									
										
										
										
											2010-05-14 21:58:08 -04:00
										 |  |  |  |     { "schema-file", 0, G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_FILENAME_ARRAY, &schema_files, NULL, NULL }, | 
					
						
							| 
									
										
										
										
											2018-06-19 23:39:24 +03:00
										 |  |  |  |     { "override-file", 0, G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_FILENAME_ARRAY, &override_files, NULL, NULL }, | 
					
						
							| 
									
										
										
										
											2021-04-28 23:22:26 +02:00
										 |  |  |  |     G_OPTION_ENTRY_NULL | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-07 10:49:29 +08:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   gchar *tmp = NULL; | 
					
						
							| 
									
										
										
										
											2023-05-23 16:49:21 +06:00
										 |  |  |  |   gchar **command_line = NULL; | 
					
						
							| 
									
										
										
										
											2011-06-07 10:49:29 +08:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-05 16:42:30 +08:00
										 |  |  |  |   setlocale (LC_ALL, GLIB_DEFAULT_LOCALE); | 
					
						
							| 
									
										
										
										
											2011-02-12 12:30:35 -05:00
										 |  |  |  |   textdomain (GETTEXT_PACKAGE); | 
					
						
							| 
									
										
										
										
											2011-06-07 10:49:29 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-20 01:23:48 +02:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							| 
									
										
										
										
											2011-06-07 10:49:29 +08:00
										 |  |  |  |   tmp = _glib_get_locale_dir (); | 
					
						
							| 
									
										
										
										
											2011-02-12 12:30:35 -05:00
										 |  |  |  |   bindtextdomain (GETTEXT_PACKAGE, tmp); | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   bindtextdomain (GETTEXT_PACKAGE, GLIB_LOCALE_DIR); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef HAVE_BIND_TEXTDOMAIN_CODESET
 | 
					
						
							|  |  |  |  |   bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8"); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-20 22:41:12 -04:00
										 |  |  |  |   context = g_option_context_new (N_("DIRECTORY")); | 
					
						
							|  |  |  |  |   g_option_context_set_translation_domain (context, GETTEXT_PACKAGE); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   g_option_context_set_summary (context, | 
					
						
							| 
									
										
										
										
											2010-04-20 22:41:12 -04:00
										 |  |  |  |     N_("Compile all GSettings schema files into a schema cache.\n" | 
					
						
							|  |  |  |  |        "Schema files are required to have the extension .gschema.xml,\n" | 
					
						
							|  |  |  |  |        "and the cache file is called gschemas.compiled.")); | 
					
						
							|  |  |  |  |   g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-23 16:49:21 +06:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   command_line = g_win32_get_command_line (); | 
					
						
							|  |  |  |  |   if (!g_option_context_parse_strv (context, &command_line, &error)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       fprintf (stderr, "%s\n", error->message); | 
					
						
							|  |  |  |  |       retval = 1; | 
					
						
							|  |  |  |  |       goto done; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   argc = g_strv_length (command_line); | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   if (!g_option_context_parse (context, &argc, &argv, &error)) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |       fprintf (stderr, "%s\n", error->message); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |       retval = 1; | 
					
						
							|  |  |  |  |       goto done; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-23 16:49:21 +06:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-30 22:56:21 +01:00
										 |  |  |  |   if (show_version_and_exit) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_print (PACKAGE_VERSION "\n"); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |       retval = 0; | 
					
						
							|  |  |  |  |       goto done; | 
					
						
							| 
									
										
										
										
											2016-09-30 22:56:21 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-21 12:14:40 +01:00
										 |  |  |  |   if (!schema_files && argc != 2) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |       fprintf (stderr, "%s\n", _("You should give exactly one directory name")); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |       retval = 1; | 
					
						
							|  |  |  |  |       goto done; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-23 16:49:21 +06:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   srcdir = command_line[1]; | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |   srcdir = argv[1]; | 
					
						
							| 
									
										
										
										
											2023-05-23 16:49:21 +06:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-20 17:27:17 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   target = g_build_filename (targetdir ? targetdir : srcdir, "gschemas.compiled", NULL); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-21 12:14:40 +01:00
										 |  |  |  |   if (!schema_files) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |       GPtrArray *overrides; | 
					
						
							| 
									
										
										
										
											2010-04-21 12:14:40 +01:00
										 |  |  |  |       GPtrArray *files; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       files = g_ptr_array_new (); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |       overrides = g_ptr_array_new (); | 
					
						
							| 
									
										
										
										
											2010-04-21 12:14:40 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  |       dir = g_dir_open (srcdir, 0, &error); | 
					
						
							|  |  |  |  |       if (dir == NULL) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           fprintf (stderr, "%s\n", error->message); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |           g_ptr_array_unref (files); | 
					
						
							|  |  |  |  |           g_ptr_array_unref (overrides); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           retval = 1; | 
					
						
							|  |  |  |  |           goto done; | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       while ((file = g_dir_read_name (dir)) != NULL) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  |           if (g_str_has_suffix (file, ".gschema.xml") || | 
					
						
							|  |  |  |  |               g_str_has_suffix (file, ".enums.xml")) | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  |             g_ptr_array_add (files, g_build_filename (srcdir, file, NULL)); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |           else if (g_str_has_suffix (file, ".gschema.override")) | 
					
						
							|  |  |  |  |             g_ptr_array_add (overrides, | 
					
						
							|  |  |  |  |                              g_build_filename (srcdir, file, NULL)); | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (files->len == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2010-06-28 14:06:32 -04:00
										 |  |  |  |           if (g_unlink (target)) | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |             fprintf (stdout, "%s\n", _("No schema files found: doing nothing.")); | 
					
						
							| 
									
										
										
										
											2010-05-19 16:02:05 -04:00
										 |  |  |  |           else | 
					
						
							| 
									
										
										
										
											2019-02-18 17:15:15 +00:00
										 |  |  |  |             fprintf (stdout, "%s\n", _("No schema files found: removed existing output file.")); | 
					
						
							| 
									
										
										
										
											2010-06-28 14:06:32 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |           g_ptr_array_unref (files); | 
					
						
							|  |  |  |  |           g_ptr_array_unref (overrides); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |           retval = 0; | 
					
						
							|  |  |  |  |           goto done; | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-12-20 00:22:05 +01:00
										 |  |  |  |       g_ptr_array_sort_values (files, compare_strings); | 
					
						
							| 
									
										
										
										
											2010-04-21 12:14:40 +01:00
										 |  |  |  |       g_ptr_array_add (files, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 00:22:05 +01:00
										 |  |  |  |       g_ptr_array_sort_values (overrides, compare_strings); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |       g_ptr_array_add (overrides, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-21 12:14:40 +01:00
										 |  |  |  |       schema_files = (char **) g_ptr_array_free (files, FALSE); | 
					
						
							| 
									
										
										
										
											2010-07-27 22:56:30 +02:00
										 |  |  |  |       override_files = (gchar **) g_ptr_array_free (overrides, FALSE); | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |   if ((table = parse_gschema_files (schema_files, strict)) == NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |       retval = 1; | 
					
						
							|  |  |  |  |       goto done; | 
					
						
							| 
									
										
										
										
											2010-09-27 10:27:11 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-04-19 15:48:30 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |   if (override_files != NULL && | 
					
						
							|  |  |  |  |       !set_overrides (table, override_files, strict)) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |       retval = 1; | 
					
						
							|  |  |  |  |       goto done; | 
					
						
							| 
									
										
										
										
											2010-09-27 11:10:11 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!dry_run && !write_to_file (table, target, &error)) | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       fprintf (stderr, "%s\n", error->message); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |       retval = 1; | 
					
						
							|  |  |  |  |       goto done; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   /* Success. */ | 
					
						
							|  |  |  |  |   retval = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | done: | 
					
						
							|  |  |  |  |   g_clear_error (&error); | 
					
						
							|  |  |  |  |   g_clear_pointer (&table, g_hash_table_unref); | 
					
						
							|  |  |  |  |   g_clear_pointer (&dir, g_dir_close); | 
					
						
							|  |  |  |  |   g_free (targetdir); | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  |   g_free (target); | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   g_strfreev (schema_files); | 
					
						
							|  |  |  |  |   g_strfreev (override_files); | 
					
						
							|  |  |  |  |   g_option_context_free (context); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   g_free (tmp); | 
					
						
							| 
									
										
										
										
											2023-05-23 16:49:21 +06:00
										 |  |  |  |   g_strfreev (command_line);   | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-20 23:28:49 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 10:02:48 +00:00
										 |  |  |  |   return retval; | 
					
						
							| 
									
										
										
										
											2010-04-15 23:35:49 -04:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-06-14 17:29:41 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Epilogue {{{1 */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* vim:set foldmethod=marker: */ |