| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  | /*
 | 
					
						
							|  |  |  |  |  * Copyright 2015 Red Hat, Inc. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2022-05-18 09:12:45 +01:00
										 |  |  |  |  * SPDX-License-Identifier: LGPL-2.1-or-later | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -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. | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -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 | 
					
						
							|  |  |  |  |  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Author: Matthias Clasen <mclasen@redhat.com> | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "config.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include <gio/gio.h>
 | 
					
						
							|  |  |  |  | #include <gi18n.h>
 | 
					
						
							|  |  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "gio-tool.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static char *attr_type = "string"; | 
					
						
							|  |  |  |  | static gboolean nofollow_symlinks = FALSE; | 
					
						
							| 
									
										
										
										
											2022-07-30 15:14:09 -05:00
										 |  |  |  | static gboolean delete = FALSE; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static const GOptionEntry entries[] = { | 
					
						
							|  |  |  |  |   { "type", 't', 0, G_OPTION_ARG_STRING, &attr_type, N_("Type of the attribute"), N_("TYPE") }, | 
					
						
							| 
									
										
										
										
											2016-09-30 05:47:15 +02:00
										 |  |  |  |   { "nofollow-symlinks", 'n', 0, G_OPTION_ARG_NONE, &nofollow_symlinks, N_("Don’t follow symbolic links"), NULL }, | 
					
						
							| 
									
										
										
										
											2022-07-30 15:14:09 -05:00
										 |  |  |  |   { "delete", 'd', 0, G_OPTION_ARG_NONE, &delete, N_("Unset given attribute"), NULL }, | 
					
						
							| 
									
										
										
										
											2021-05-13 20:16:46 +00:00
										 |  |  |  |   G_OPTION_ENTRY_NULL | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static char * | 
					
						
							|  |  |  |  | hex_unescape (const char *str) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-04-25 00:37:47 +01:00
										 |  |  |  |   size_t i; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |   char *unescaped_str, *p; | 
					
						
							|  |  |  |  |   unsigned char c; | 
					
						
							| 
									
										
										
										
											2024-04-25 00:37:47 +01:00
										 |  |  |  |   size_t len; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   len = strlen (str); | 
					
						
							|  |  |  |  |   unescaped_str = g_malloc (len + 1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   p = unescaped_str; | 
					
						
							|  |  |  |  |   for (i = 0; i < len; i++) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (str[i] == '\\' && | 
					
						
							|  |  |  |  | 	  str[i+1] == 'x' && | 
					
						
							|  |  |  |  | 	  len - i >= 4) | 
					
						
							|  |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  c = | 
					
						
							|  |  |  |  | 	    (g_ascii_xdigit_value (str[i+2]) << 4) | | 
					
						
							|  |  |  |  | 	    g_ascii_xdigit_value (str[i+3]); | 
					
						
							|  |  |  |  | 	  *p++ = c; | 
					
						
							|  |  |  |  | 	  i += 3; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  | 	*p++ = str[i]; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   *p++ = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return unescaped_str; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int | 
					
						
							|  |  |  |  | handle_set (int argc, char *argv[], gboolean do_help) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GOptionContext *context; | 
					
						
							|  |  |  |  |   GError *error = NULL; | 
					
						
							|  |  |  |  |   GFile *file; | 
					
						
							|  |  |  |  |   const char *attribute; | 
					
						
							|  |  |  |  |   GFileAttributeType type; | 
					
						
							|  |  |  |  |   gpointer value; | 
					
						
							| 
									
										
										
										
											2022-04-27 15:01:08 +01:00
										 |  |  |  |   gpointer value_allocated = NULL; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |   gboolean b; | 
					
						
							|  |  |  |  |   guint32 uint32; | 
					
						
							|  |  |  |  |   gint32 int32; | 
					
						
							|  |  |  |  |   guint64 uint64; | 
					
						
							|  |  |  |  |   gint64 int64; | 
					
						
							|  |  |  |  |   gchar *param; | 
					
						
							| 
									
										
										
										
											2022-04-27 15:01:08 +01:00
										 |  |  |  |   int retval = 0; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_set_prgname ("gio set"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* Translators: commandline placeholder */ | 
					
						
							| 
									
										
										
										
											2018-05-25 16:59:30 +02:00
										 |  |  |  |   param = g_strdup_printf ("%s %s %s…", _("LOCATION"), _("ATTRIBUTE"), _("VALUE")); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |   context = g_option_context_new (param); | 
					
						
							|  |  |  |  |   g_free (param); | 
					
						
							|  |  |  |  |   g_option_context_set_help_enabled (context, FALSE); | 
					
						
							|  |  |  |  |   g_option_context_set_summary (context, _("Set a file attribute of LOCATION.")); | 
					
						
							|  |  |  |  |   g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (do_help) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       show_help (context, NULL); | 
					
						
							| 
									
										
										
										
											2016-12-19 12:11:13 +01:00
										 |  |  |  |       g_option_context_free (context); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       return 0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!g_option_context_parse (context, &argc, &argv, &error)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       show_help (context, error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2016-12-19 12:11:13 +01:00
										 |  |  |  |       g_option_context_free (context); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       return 1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (argc < 2) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       show_help (context, _("Location not specified")); | 
					
						
							| 
									
										
										
										
											2016-12-19 12:11:13 +01:00
										 |  |  |  |       g_option_context_free (context); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       return 1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (argc < 3) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       show_help (context, _("Attribute not specified")); | 
					
						
							| 
									
										
										
										
											2016-12-19 12:11:13 +01:00
										 |  |  |  |       g_option_context_free (context); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       return 1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   attribute = argv[2]; | 
					
						
							| 
									
										
										
										
											2022-07-30 15:14:09 -05:00
										 |  |  |  |   if (delete) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       type = G_FILE_ATTRIBUTE_TYPE_INVALID; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       type = attribute_type_from_string (attr_type); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if ((argc < 4) && (type != G_FILE_ATTRIBUTE_TYPE_INVALID)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       show_help (context, _("Value not specified")); | 
					
						
							| 
									
										
										
										
											2016-12-19 12:11:13 +01:00
										 |  |  |  |       g_option_context_free (context); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       return 1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if ((argc > 4) && (type != G_FILE_ATTRIBUTE_TYPE_STRINGV)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       show_help (context, _("Too many arguments")); | 
					
						
							| 
									
										
										
										
											2016-12-19 12:11:13 +01:00
										 |  |  |  |       g_option_context_free (context); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       return 1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_option_context_free (context); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   switch (type) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_STRING: | 
					
						
							|  |  |  |  |       value = argv[3]; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_BYTE_STRING: | 
					
						
							| 
									
										
										
										
											2022-04-27 15:01:08 +01:00
										 |  |  |  |       value = value_allocated = hex_unescape (argv[3]); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |       b = g_ascii_strcasecmp (argv[3], "true") == 0; | 
					
						
							|  |  |  |  |       value = &b; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_UINT32: | 
					
						
							|  |  |  |  |       uint32 = atol (argv[3]); | 
					
						
							|  |  |  |  |       value = &uint32; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_INT32: | 
					
						
							|  |  |  |  |       int32 = atol (argv[3]); | 
					
						
							|  |  |  |  |       value = &int32; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_UINT64: | 
					
						
							|  |  |  |  |       uint64 = g_ascii_strtoull (argv[3], NULL, 10); | 
					
						
							|  |  |  |  |       value = &uint64; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_INT64: | 
					
						
							|  |  |  |  |       int64 = g_ascii_strtoll (argv[3], NULL, 10); | 
					
						
							|  |  |  |  |       value = &int64; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_STRINGV: | 
					
						
							|  |  |  |  |       value = &argv[3]; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_INVALID: | 
					
						
							|  |  |  |  |       value = NULL; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case G_FILE_ATTRIBUTE_TYPE_OBJECT: | 
					
						
							|  |  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2017-04-10 15:40:42 +02:00
										 |  |  |  |       print_error (_("Invalid attribute type “%s”"), attr_type); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       return 1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-16 14:35:55 +01:00
										 |  |  |  |   file = g_file_new_for_commandline_arg (argv[1]); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |   if (!g_file_set_attribute (file, | 
					
						
							|  |  |  |  | 			     attribute, | 
					
						
							|  |  |  |  | 			     type, | 
					
						
							|  |  |  |  | 			     value, | 
					
						
							|  |  |  |  |                              nofollow_symlinks ? | 
					
						
							|  |  |  |  |                                G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS : | 
					
						
							|  |  |  |  |                                G_FILE_QUERY_INFO_NONE, | 
					
						
							|  |  |  |  |                              NULL, &error)) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-04-10 13:00:44 +02:00
										 |  |  |  |       print_error ("%s", error->message); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2022-04-27 15:01:08 +01:00
										 |  |  |  |       retval = 1; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 15:01:08 +01:00
										 |  |  |  |   g_clear_pointer (&value_allocated, g_free); | 
					
						
							| 
									
										
										
										
											2016-12-16 14:35:55 +01:00
										 |  |  |  |   g_object_unref (file); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 15:01:08 +01:00
										 |  |  |  |   return retval; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |  | } |