| 
									
										
										
										
											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>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-04 12:27:42 +08:00
										 |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  | #include <io.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef STDIN_FILENO
 | 
					
						
							|  |  |  | #define STDIN_FILENO 0
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-04 12:27:42 +08:00
										 |  |  | #ifdef HAVE_UNISTD_H
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "gio-tool.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-19 17:27:34 +01:00
										 |  |  | static char *global_etag = NULL; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  | static gboolean backup = FALSE; | 
					
						
							|  |  |  | static gboolean create = FALSE; | 
					
						
							|  |  |  | static gboolean append = FALSE; | 
					
						
							|  |  |  | static gboolean priv = FALSE; | 
					
						
							|  |  |  | static gboolean replace_dest = FALSE; | 
					
						
							|  |  |  | static gboolean print_etag = FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const GOptionEntry entries[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   { "backup", 'b', 0, G_OPTION_ARG_NONE, &backup, N_("Backup existing destination files"), NULL }, | 
					
						
							|  |  |  |   { "create", 'c', 0, G_OPTION_ARG_NONE, &create, N_("Only create if not existing"), NULL }, | 
					
						
							|  |  |  |   { "append", 'a', 0, G_OPTION_ARG_NONE, &append, N_("Append to end of file"), NULL }, | 
					
						
							|  |  |  |   { "private", 'p', 0, G_OPTION_ARG_NONE, &priv, N_("When creating, restrict access to the current user"), NULL }, | 
					
						
							|  |  |  |   { "unlink", 'u', 0, G_OPTION_ARG_NONE, &replace_dest, N_("When replacing, replace as if the destination did not exist"), NULL }, | 
					
						
							|  |  |  |   /* Translators: The "etag" is a token allowing to verify whether a file has been modified */ | 
					
						
							|  |  |  |   { "print-etag", 'v', 0, G_OPTION_ARG_NONE, &print_etag, N_("Print new etag at end"), NULL }, | 
					
						
							|  |  |  |   /* Translators: The "etag" is a token allowing to verify whether a file has been modified */ | 
					
						
							| 
									
										
										
										
											2022-01-19 17:27:34 +01:00
										 |  |  |   { "etag", 'e', 0, G_OPTION_ARG_STRING, &global_etag, N_("The etag of the file being overwritten"), N_("ETAG") }, | 
					
						
							| 
									
										
										
										
											2021-05-13 20:16:46 +00:00
										 |  |  |   G_OPTION_ENTRY_NULL | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-11 13:21:28 +02:00
										 |  |  | /* 256k minus malloc overhead */ | 
					
						
							|  |  |  | #define STREAM_BUFFER_SIZE (1024*256 - 2*sizeof(gpointer))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  | static gboolean | 
					
						
							|  |  |  | save (GFile *file) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GOutputStream *out; | 
					
						
							|  |  |  |   GFileCreateFlags flags; | 
					
						
							| 
									
										
										
										
											2017-08-11 13:21:28 +02:00
										 |  |  |   char *buffer; | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |   gssize res; | 
					
						
							|  |  |  |   gboolean close_res; | 
					
						
							|  |  |  |   GError *error; | 
					
						
							|  |  |  |   gboolean save_res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   error = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   flags = priv ? G_FILE_CREATE_PRIVATE : G_FILE_CREATE_NONE; | 
					
						
							|  |  |  |   flags |= replace_dest ? G_FILE_CREATE_REPLACE_DESTINATION : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (create) | 
					
						
							|  |  |  |     out = (GOutputStream *)g_file_create (file, flags, NULL, &error); | 
					
						
							|  |  |  |   else if (append) | 
					
						
							|  |  |  |     out = (GOutputStream *)g_file_append_to (file, flags, NULL, &error); | 
					
						
							|  |  |  |   else | 
					
						
							| 
									
										
										
										
											2022-01-19 17:27:34 +01:00
										 |  |  |     out = (GOutputStream *)g_file_replace (file, global_etag, backup, flags, NULL, &error); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |   if (out == NULL) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       print_file_error (file, error->message); | 
					
						
							|  |  |  |       g_error_free (error); | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-11 13:21:28 +02:00
										 |  |  |   buffer = g_malloc (STREAM_BUFFER_SIZE); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |   save_res = TRUE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (1) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-08-11 13:21:28 +02:00
										 |  |  |       res = read (STDIN_FILENO, buffer, STREAM_BUFFER_SIZE); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |       if (res > 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2017-08-11 13:47:14 +02:00
										 |  |  |           g_output_stream_write_all (out, buffer, res, NULL, NULL, &error); | 
					
						
							|  |  |  |           if (error != NULL) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               save_res = FALSE; | 
					
						
							| 
									
										
										
										
											2017-08-11 13:50:36 +02:00
										 |  |  |               print_file_error (file, error->message); | 
					
						
							| 
									
										
										
										
											2017-08-11 13:47:14 +02:00
										 |  |  |               g_clear_error (&error); | 
					
						
							|  |  |  |               goto out; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  |       else if (res < 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  save_res = FALSE; | 
					
						
							| 
									
										
										
										
											2017-04-10 13:00:44 +02:00
										 |  |  |           print_error ("%s", _("Error reading from standard input")); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (res == 0) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  out: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   close_res = g_output_stream_close (out, NULL, &error); | 
					
						
							|  |  |  |   if (!close_res) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       save_res = FALSE; | 
					
						
							| 
									
										
										
										
											2017-08-11 13:50:36 +02:00
										 |  |  |       print_file_error (file, error->message); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |       g_error_free (error); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (close_res && print_etag) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       char *etag; | 
					
						
							|  |  |  |       etag = g_file_output_stream_get_etag (G_FILE_OUTPUT_STREAM (out)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (etag) | 
					
						
							|  |  |  | 	g_print ("Etag: %s\n", etag); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	/* Translators: The "etag" is a token allowing to verify whether a file has been modified */ | 
					
						
							|  |  |  | 	g_print (_("Etag not available\n")); | 
					
						
							|  |  |  |       g_free (etag); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_object_unref (out); | 
					
						
							| 
									
										
										
										
											2017-08-11 13:21:28 +02:00
										 |  |  |   g_free (buffer); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return save_res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | handle_save (int argc, char *argv[], gboolean do_help) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GOptionContext *context; | 
					
						
							|  |  |  |   GError *error = NULL; | 
					
						
							|  |  |  |   GFile *file; | 
					
						
							|  |  |  |   gboolean res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_set_prgname ("gio save"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Translators: commandline placeholder */ | 
					
						
							|  |  |  |   context = g_option_context_new (_("DESTINATION")); | 
					
						
							|  |  |  |   g_option_context_set_help_enabled (context, FALSE); | 
					
						
							|  |  |  |   g_option_context_set_summary (context, | 
					
						
							|  |  |  |       _("Read from standard input and save to DEST.")); | 
					
						
							|  |  |  |   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, _("No destination given")); | 
					
						
							| 
									
										
										
										
											2016-12-19 12:11:13 +01:00
										 |  |  |       g_option_context_free (context); | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |       return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-11 21:47:49 +01:00
										 |  |  |   if (argc > 2) | 
					
						
							| 
									
										
										
										
											2015-05-25 13:29:02 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |       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); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   file = g_file_new_for_commandline_arg (argv[1]); | 
					
						
							|  |  |  |   res = save (file); | 
					
						
							|  |  |  |   g_object_unref (file); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return res ? 0 : 2; | 
					
						
							|  |  |  | } |