1 /* GIO - GLib Input, Output and Streaming Library
2 *
3 * Copyright (C) 2009 Red Hat, Inc.
4 *
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
9 *
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
14 *
15 * You should have received a copy of the GNU Lesser General
16 * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
17 *
18 * Author: Alexander Larsson <alexl@redhat.com>
19 */
20
21 #include <config.h>
22
23 #include <stdio.h>
24 #include <locale.h>
25 #include <errno.h>
26
27 #include <glib.h>
28 #include <gio/gio.h>
29
30 #ifdef G_OS_UNIX
31 #include <unistd.h>
32 #endif
33
34 #ifdef G_OS_WIN32
35 #include <io.h>
36 #ifndef STDOUT_FILENO
37 #define STDOUT_FILENO 1
38 #endif
39 #endif
40
41 static gchar **locations = NULL;
42 static char *from_charset = NULL;
43 static char *to_charset = NULL;
44 static gboolean decompress = FALSE;
45 static gboolean compress = FALSE;
46 static gboolean gzip = FALSE;
47 static gboolean fallback = FALSE;
48
49 static GOptionEntry entries[] = {
50 {"decompress", 0, 0, G_OPTION_ARG_NONE, &decompress, "decompress", NULL},
51 {"compress", 0, 0, G_OPTION_ARG_NONE, &compress, "compress", NULL},
52 {"gzip", 0, 0, G_OPTION_ARG_NONE, &gzip, "use gzip format", NULL},
53 {"from-charset", 0, 0, G_OPTION_ARG_STRING, &from_charset, "from charset", NULL},
54 {"to-charset", 0, 0, G_OPTION_ARG_STRING, &to_charset, "to charset", NULL},
55 {"fallback", 0, 0, G_OPTION_ARG_NONE, &fallback, "use fallback", NULL},
56 {G_OPTION_REMAINING, 0, 0, G_OPTION_ARG_FILENAME_ARRAY, &locations, "locations", NULL},
57 {NULL}
58 };
59
60 static void
decompressor_file_info_notify_cb(GZlibDecompressor * decompressor,GParamSpec * pspec,gpointer data)61 decompressor_file_info_notify_cb (GZlibDecompressor *decompressor,
62 GParamSpec *pspec,
63 gpointer data)
64 {
65 GFileInfo *file_info;
66 const gchar *filename;
67
68 file_info = g_zlib_decompressor_get_file_info (decompressor);
69 if (file_info == NULL)
70 return;
71
72 filename = g_file_info_get_name (file_info);
73 if (filename)
74 g_printerr ("Decompressor filename: %s\n", filename);
75 }
76
77 static void
cat(GFile * file)78 cat (GFile * file)
79 {
80 GInputStream *in;
81 char buffer[1024 * 8 + 1];
82 char *p;
83 gssize res;
84 gboolean close_res;
85 GError *error;
86 GConverter *conv;
87 GCharsetConverter *cconv = NULL;
88
89 error = NULL;
90 in = (GInputStream *) g_file_read (file, NULL, &error);
91 if (in == NULL)
92 {
93 /* Translators: the first %s is the program name, the second one */
94 /* is the URI of the file, the third is the error message. */
95 g_printerr ("%s: %s: error opening file: %s\n",
96 g_get_prgname (), g_file_get_uri (file), error->message);
97 g_error_free (error);
98 return;
99 }
100
101 if (decompress)
102 {
103 GInputStream *old;
104 conv = (GConverter *)g_zlib_decompressor_new (gzip?G_ZLIB_COMPRESSOR_FORMAT_GZIP:G_ZLIB_COMPRESSOR_FORMAT_ZLIB);
105 old = in;
106 in = (GInputStream *) g_converter_input_stream_new (in, conv);
107 g_signal_connect (conv, "notify::file-info", G_CALLBACK (decompressor_file_info_notify_cb), NULL);
108 g_object_unref (conv);
109 g_object_unref (old);
110 }
111
112 if (from_charset && to_charset)
113 {
114 cconv = g_charset_converter_new (to_charset, from_charset, &error);
115 conv = (GConverter *)cconv;
116 if (conv)
117 {
118 GInputStream *old;
119
120 g_charset_converter_set_use_fallback (cconv, fallback);
121
122 old = in;
123 in = (GInputStream *) g_converter_input_stream_new (in, conv);
124 g_object_unref (conv);
125 g_object_unref (old);
126 }
127 else
128 {
129 g_printerr ("%s: Can't convert between charsets: %s\n",
130 g_get_prgname (), error->message);
131 }
132 }
133
134 if (compress)
135 {
136 GInputStream *old;
137 GFileInfo *in_file_info;
138
139 in_file_info = g_file_query_info (file,
140 G_FILE_ATTRIBUTE_STANDARD_NAME ","
141 G_FILE_ATTRIBUTE_TIME_MODIFIED,
142 G_FILE_QUERY_INFO_NONE,
143 NULL,
144 &error);
145 if (in_file_info == NULL)
146 {
147 g_printerr ("%s: %s: error reading file info: %s\n",
148 g_get_prgname (), g_file_get_uri (file), error->message);
149 g_error_free (error);
150 return;
151 }
152
153 conv = (GConverter *)g_zlib_compressor_new(gzip?G_ZLIB_COMPRESSOR_FORMAT_GZIP:G_ZLIB_COMPRESSOR_FORMAT_ZLIB, -1);
154 g_zlib_compressor_set_file_info (G_ZLIB_COMPRESSOR (conv), in_file_info);
155 old = in;
156 in = (GInputStream *) g_converter_input_stream_new (in, conv);
157 g_object_unref (conv);
158 g_object_unref (old);
159 g_object_unref (in_file_info);
160 }
161
162 while (1)
163 {
164 res =
165 g_input_stream_read (in, buffer, sizeof (buffer) - 1, NULL, &error);
166 if (res > 0)
167 {
168 gssize written;
169
170 p = buffer;
171 while (res > 0)
172 {
173 written = write (STDOUT_FILENO, p, res);
174
175 if (written == -1 && errno != EINTR)
176 {
177 /* Translators: the first %s is the program name, the */
178 /* second one is the URI of the file. */
179 g_printerr ("%s: %s, error writing to stdout",
180 g_get_prgname (), g_file_get_uri (file));
181 goto out;
182 }
183 res -= written;
184 p += written;
185 }
186 }
187 else if (res < 0)
188 {
189 g_printerr ("%s: %s: error reading: %s\n",
190 g_get_prgname (), g_file_get_uri (file),
191 error->message);
192 g_error_free (error);
193 error = NULL;
194 break;
195 }
196 else if (res == 0)
197 break;
198 }
199
200 out:
201
202 close_res = g_input_stream_close (in, NULL, &error);
203 if (!close_res)
204 {
205 g_printerr ("%s: %s:error closing: %s\n",
206 g_get_prgname (), g_file_get_uri (file), error->message);
207 g_error_free (error);
208 }
209
210 if (cconv != NULL && fallback)
211 {
212 guint num = g_charset_converter_get_num_fallbacks (cconv);
213 if (num > 0)
214 g_printerr ("Number of fallback errors: %u\n", num);
215 }
216 }
217
218 int
main(int argc,char * argv[])219 main (int argc, char *argv[])
220 {
221 GError *error = NULL;
222 GOptionContext *context = NULL;
223 GFile *file;
224 int i;
225
226 context =
227 g_option_context_new ("LOCATION... - concatenate LOCATIONS "
228 "to standard output.");
229
230 g_option_context_set_summary (context, "filter files");
231
232 g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE);
233 g_option_context_parse (context, &argc, &argv, &error);
234
235 g_option_context_free (context);
236
237 if (error != NULL)
238 {
239 g_printerr ("Error parsing commandline options: %s\n", error->message);
240 g_printerr ("\n");
241 g_printerr ("Try \"%s --help\" for more information.",
242 g_get_prgname ());
243 g_printerr ("\n");
244 g_error_free(error);
245 return 1;
246 }
247
248 if (!locations)
249 {
250 g_printerr ("%s: missing locations", g_get_prgname ());
251 g_printerr ("\n");
252 g_printerr ("Try \"%s --help\" for more information.",
253 g_get_prgname ());
254 g_printerr ("\n");
255 return 1;
256 }
257
258 i = 0;
259
260 do
261 {
262 file = g_file_new_for_commandline_arg (locations[i]);
263 cat (file);
264 g_object_unref (file);
265 }
266 while (locations[++i] != NULL);
267
268 return 0;
269 }
270