Whitespace cleanups.
[platform/upstream/glib.git] / gio / glocalfileoutputstream.c
1 /* GIO - GLib Input, Output and Streaming Library
2  * 
3  * Copyright (C) 2006-2007 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 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, write to the
17  * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
18  * Boston, MA 02111-1307, USA.
19  *
20  * Author: Alexander Larsson <alexl@redhat.com>
21  */
22
23 #include <config.h>
24
25 #include <sys/types.h>
26 #include <sys/stat.h>
27 #include <fcntl.h>
28 #ifdef HAVE_UNISTD_H
29 #include <unistd.h>
30 #endif
31 #include <errno.h>
32 #include <string.h>
33
34 #include <glib.h>
35 #include <glib/gstdio.h>
36 #include "glibintl.h"
37 #include "gioerror.h"
38 #include "glocalfileoutputstream.h"
39 #include "glocalfileinfo.h"
40
41 #ifdef G_OS_WIN32
42 #include <io.h>
43 #ifndef S_ISDIR
44 #define S_ISDIR(m) (((m) & _S_IFMT) == _S_IFDIR)
45 #endif
46 #ifndef S_ISREG
47 #define S_ISREG(m) (((m) & _S_IFMT) == _S_IFREG)
48 #endif
49 #endif
50
51 #include "gioalias.h"
52
53 #define g_local_file_output_stream_get_type _g_local_file_output_stream_get_type
54 G_DEFINE_TYPE (GLocalFileOutputStream, g_local_file_output_stream, G_TYPE_FILE_OUTPUT_STREAM);
55
56 /* Some of the file replacement code was based on the code from gedit,
57  * relicenced to LGPL with permissions from the authors.
58  */
59
60 #define BACKUP_EXTENSION "~"
61
62 struct _GLocalFileOutputStreamPrivate {
63   char *tmp_filename;
64   char *original_filename;
65   char *backup_filename;
66   char *etag;
67   int fd;
68 };
69
70 static gssize     g_local_file_output_stream_write        (GOutputStream      *stream,
71                                                            const void         *buffer,
72                                                            gsize               count,
73                                                            GCancellable       *cancellable,
74                                                            GError            **error);
75 static gboolean   g_local_file_output_stream_close        (GOutputStream      *stream,
76                                                            GCancellable       *cancellable,
77                                                            GError            **error);
78 static GFileInfo *g_local_file_output_stream_query_info   (GFileOutputStream  *stream,
79                                                            char               *attributes,
80                                                            GCancellable       *cancellable,
81                                                            GError            **error);
82 static char *     g_local_file_output_stream_get_etag     (GFileOutputStream  *stream);
83 static goffset    g_local_file_output_stream_tell         (GFileOutputStream  *stream);
84 static gboolean   g_local_file_output_stream_can_seek     (GFileOutputStream  *stream);
85 static gboolean   g_local_file_output_stream_seek         (GFileOutputStream  *stream,
86                                                            goffset             offset,
87                                                            GSeekType           type,
88                                                            GCancellable       *cancellable,
89                                                            GError            **error);
90 static gboolean   g_local_file_output_stream_can_truncate (GFileOutputStream  *stream);
91 static gboolean   g_local_file_output_stream_truncate     (GFileOutputStream  *stream,
92                                                            goffset             size,
93                                                            GCancellable       *cancellable,
94                                                            GError            **error);
95
96 static void
97 g_local_file_output_stream_finalize (GObject *object)
98 {
99   GLocalFileOutputStream *file;
100   
101   file = G_LOCAL_FILE_OUTPUT_STREAM (object);
102   
103   g_free (file->priv->tmp_filename);
104   g_free (file->priv->original_filename);
105   g_free (file->priv->backup_filename);
106   g_free (file->priv->etag);
107   
108   if (G_OBJECT_CLASS (g_local_file_output_stream_parent_class)->finalize)
109     (*G_OBJECT_CLASS (g_local_file_output_stream_parent_class)->finalize) (object);
110 }
111
112 static void
113 g_local_file_output_stream_class_init (GLocalFileOutputStreamClass *klass)
114 {
115   GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
116   GOutputStreamClass *stream_class = G_OUTPUT_STREAM_CLASS (klass);
117   GFileOutputStreamClass *file_stream_class = G_FILE_OUTPUT_STREAM_CLASS (klass);
118   
119   g_type_class_add_private (klass, sizeof (GLocalFileOutputStreamPrivate));
120   
121   gobject_class->finalize = g_local_file_output_stream_finalize;
122
123   stream_class->write_fn = g_local_file_output_stream_write;
124   stream_class->close_fn = g_local_file_output_stream_close;
125   file_stream_class->query_info = g_local_file_output_stream_query_info;
126   file_stream_class->get_etag = g_local_file_output_stream_get_etag;
127   file_stream_class->tell = g_local_file_output_stream_tell;
128   file_stream_class->can_seek = g_local_file_output_stream_can_seek;
129   file_stream_class->seek = g_local_file_output_stream_seek;
130   file_stream_class->can_truncate = g_local_file_output_stream_can_truncate;
131   file_stream_class->truncate_fn = g_local_file_output_stream_truncate;
132 }
133
134 static void
135 g_local_file_output_stream_init (GLocalFileOutputStream *stream)
136 {
137   stream->priv = G_TYPE_INSTANCE_GET_PRIVATE (stream,
138                                               G_TYPE_LOCAL_FILE_OUTPUT_STREAM,
139                                               GLocalFileOutputStreamPrivate);
140 }
141
142 static gssize
143 g_local_file_output_stream_write (GOutputStream  *stream,
144                                   const void     *buffer,
145                                   gsize           count,
146                                   GCancellable   *cancellable,
147                                   GError        **error)
148 {
149   GLocalFileOutputStream *file;
150   gssize res;
151
152   file = G_LOCAL_FILE_OUTPUT_STREAM (stream);
153
154   while (1)
155     {
156       if (g_cancellable_set_error_if_cancelled (cancellable, error))
157         return -1;
158       res = write (file->priv->fd, buffer, count);
159       if (res == -1)
160         {
161           if (errno == EINTR)
162             continue;
163           
164           g_set_error (error, G_IO_ERROR,
165                        g_io_error_from_errno (errno),
166                        _("Error writing to file: %s"),
167                        g_strerror (errno));
168         }
169       
170       break;
171     }
172   
173   return res;
174 }
175
176 static gboolean
177 g_local_file_output_stream_close (GOutputStream  *stream,
178                                   GCancellable   *cancellable,
179                                   GError        **error)
180 {
181   GLocalFileOutputStream *file;
182   struct stat final_stat;
183   int res;
184
185   file = G_LOCAL_FILE_OUTPUT_STREAM (stream);
186
187   if (file->priv->tmp_filename)
188     {
189       /* We need to move the temp file to its final place,
190        * and possibly create the backup file
191        */
192
193       if (file->priv->backup_filename)
194         {
195           if (g_cancellable_set_error_if_cancelled (cancellable, error))
196             goto err_out;
197           
198 #ifdef HAVE_LINK
199           /* create original -> backup link, the original is then renamed over */
200           if (unlink (file->priv->backup_filename) != 0 &&
201               errno != ENOENT)
202             {
203               g_set_error (error, G_IO_ERROR,
204                            G_IO_ERROR_CANT_CREATE_BACKUP,
205                            _("Error removing old backup link: %s"),
206                            g_strerror (errno));
207               goto err_out;
208             }
209
210           if (link (file->priv->original_filename, file->priv->backup_filename) != 0)
211             {
212               g_set_error (error, G_IO_ERROR,
213                            G_IO_ERROR_CANT_CREATE_BACKUP,
214                            _("Error creating backup link: %s"),
215                            g_strerror (errno));
216               goto err_out;
217             }
218 #else
219             /* If link not supported, just rename... */
220           if (!rename (file->priv->original_filename, file->priv->backup_filename) != 0)
221             {
222               g_set_error (error, G_IO_ERROR,
223                            G_IO_ERROR_CANT_CREATE_BACKUP,
224                            _("Error creating backup copy: %s"),
225                            g_strerror (errno));
226               goto err_out;
227             }
228 #endif
229         }
230       
231
232       if (g_cancellable_set_error_if_cancelled (cancellable, error))
233         goto err_out;
234
235       /* tmp -> original */
236       if (rename (file->priv->tmp_filename, file->priv->original_filename) != 0)
237         {
238           g_set_error (error, G_IO_ERROR,
239                        g_io_error_from_errno (errno),
240                        _("Error renaming temporary file: %s"),
241                        g_strerror (errno));
242           goto err_out;
243         }
244     }
245   
246   if (g_cancellable_set_error_if_cancelled (cancellable, error))
247     goto err_out;
248       
249   if (fstat (file->priv->fd, &final_stat) == 0)
250     file->priv->etag = _g_local_file_info_create_etag (&final_stat);
251
252   while (1)
253     {
254       res = close (file->priv->fd);
255       if (res == -1)
256         {
257           g_set_error (error, G_IO_ERROR,
258                        g_io_error_from_errno (errno),
259                        _("Error closing file: %s"),
260                        g_strerror (errno));
261         }
262       break;
263     }
264   
265   return res != -1;
266
267  err_out:
268   /* A simple try to close the fd in case we fail before the actual close */
269   close (file->priv->fd);
270   return FALSE;
271 }
272
273 static char *
274 g_local_file_output_stream_get_etag (GFileOutputStream *stream)
275 {
276   GLocalFileOutputStream *file;
277
278   file = G_LOCAL_FILE_OUTPUT_STREAM (stream);
279   
280   return g_strdup (file->priv->etag);
281 }
282
283 static goffset
284 g_local_file_output_stream_tell (GFileOutputStream *stream)
285 {
286   GLocalFileOutputStream *file;
287   off_t pos;
288
289   file = G_LOCAL_FILE_OUTPUT_STREAM (stream);
290   
291   pos = lseek (file->priv->fd, 0, SEEK_CUR);
292
293   if (pos == (off_t)-1)
294     return 0;
295   
296   return pos;
297 }
298
299 static gboolean
300 g_local_file_output_stream_can_seek (GFileOutputStream *stream)
301 {
302   GLocalFileOutputStream *file;
303   off_t pos;
304
305   file = G_LOCAL_FILE_OUTPUT_STREAM (stream);
306   
307   pos = lseek (file->priv->fd, 0, SEEK_CUR);
308
309   if (pos == (off_t)-1 && errno == ESPIPE)
310     return FALSE;
311   
312   return TRUE;
313 }
314
315 static int
316 seek_type_to_lseek (GSeekType type)
317 {
318   switch (type)
319     {
320     default:
321     case G_SEEK_CUR:
322       return SEEK_CUR;
323       
324     case G_SEEK_SET:
325       return SEEK_SET;
326       
327     case G_SEEK_END:
328       return SEEK_END;
329     }
330 }
331
332 static gboolean
333 g_local_file_output_stream_seek (GFileOutputStream  *stream,
334                                  goffset             offset,
335                                  GSeekType           type,
336                                  GCancellable       *cancellable,
337                                  GError            **error)
338 {
339   GLocalFileOutputStream *file;
340   off_t pos;
341
342   file = G_LOCAL_FILE_OUTPUT_STREAM (stream);
343
344   pos = lseek (file->priv->fd, offset, seek_type_to_lseek (type));
345
346   if (pos == (off_t)-1)
347     {
348       g_set_error (error, G_IO_ERROR,
349                    g_io_error_from_errno (errno),
350                    _("Error seeking in file: %s"),
351                    g_strerror (errno));
352       return FALSE;
353     }
354   
355   return TRUE;
356 }
357
358 static gboolean
359 g_local_file_output_stream_can_truncate (GFileOutputStream *stream)
360 {
361   /* We can't truncate pipes and stuff where we can't seek */
362   return g_local_file_output_stream_can_seek (stream);
363 }
364
365 static gboolean
366 g_local_file_output_stream_truncate (GFileOutputStream  *stream,
367                                      goffset             size,
368                                      GCancellable       *cancellable,
369                                      GError            **error)
370 {
371   GLocalFileOutputStream *file;
372   int res;
373
374   file = G_LOCAL_FILE_OUTPUT_STREAM (stream);
375
376  restart:
377 #ifdef G_OS_WIN32
378   res = g_win32_ftruncate (file->priv->fd, size);
379 #else
380   res = ftruncate (file->priv->fd, size);
381 #endif
382   
383   if (res == -1)
384     {
385       if (errno == EINTR)
386         {
387           if (g_cancellable_set_error_if_cancelled (cancellable, error))
388             return FALSE;
389           goto restart;
390         }
391
392       g_set_error (error, G_IO_ERROR,
393                    g_io_error_from_errno (errno),
394                    _("Error truncating file: %s"),
395                    g_strerror (errno));
396       return FALSE;
397     }
398   
399   return TRUE;
400 }
401
402
403 static GFileInfo *
404 g_local_file_output_stream_query_info (GFileOutputStream  *stream,
405                                        char               *attributes,
406                                        GCancellable       *cancellable,
407                                        GError            **error)
408 {
409   GLocalFileOutputStream *file;
410
411   file = G_LOCAL_FILE_OUTPUT_STREAM (stream);
412
413   if (g_cancellable_set_error_if_cancelled (cancellable, error))
414     return NULL;
415   
416   return _g_local_file_info_get_from_fd (file->priv->fd,
417                                          attributes,
418                                          error);
419 }
420
421 GFileOutputStream *
422 _g_local_file_output_stream_create  (const char        *filename,
423                                      GFileCreateFlags   flags,
424                                      GCancellable      *cancellable,
425                                      GError           **error)
426 {
427   GLocalFileOutputStream *stream;
428   int mode;
429   int fd;
430
431   if (g_cancellable_set_error_if_cancelled (cancellable, error))
432     return NULL;
433
434   if (flags & G_FILE_CREATE_PRIVATE)
435     mode = 0600;
436   else
437     mode = 0666;
438   
439   fd = g_open (filename, O_CREAT | O_EXCL | O_WRONLY, mode);
440   if (fd == -1)
441     {
442       int errsv = errno;
443
444       if (errsv == EINVAL)
445         /* This must be an invalid filename, on e.g. FAT */
446         g_set_error (error, G_IO_ERROR,
447                      G_IO_ERROR_INVALID_FILENAME,
448                      _("Invalid filename"));
449       else
450         g_set_error (error, G_IO_ERROR,
451                      g_io_error_from_errno (errsv),
452                      _("Error opening file '%s': %s"),
453                      filename, g_strerror (errsv));
454       return NULL;
455     }
456   
457   stream = g_object_new (G_TYPE_LOCAL_FILE_OUTPUT_STREAM, NULL);
458   stream->priv->fd = fd;
459   return G_FILE_OUTPUT_STREAM (stream);
460 }
461
462 GFileOutputStream *
463 _g_local_file_output_stream_append  (const char        *filename,
464                                      GFileCreateFlags   flags,
465                                      GCancellable      *cancellable,
466                                      GError           **error)
467 {
468   GLocalFileOutputStream *stream;
469   int mode;
470   int fd;
471
472   if (g_cancellable_set_error_if_cancelled (cancellable, error))
473     return NULL;
474
475   if (flags & G_FILE_CREATE_PRIVATE)
476     mode = 0600;
477   else
478     mode = 0666;
479
480   fd = g_open (filename, O_CREAT | O_APPEND | O_WRONLY, mode);
481   if (fd == -1)
482     {
483       int errsv = errno;
484
485       if (errsv == EINVAL)
486         /* This must be an invalid filename, on e.g. FAT */
487         g_set_error (error, G_IO_ERROR,
488                      G_IO_ERROR_INVALID_FILENAME,
489                      _("Invalid filename"));
490       else
491         g_set_error (error, G_IO_ERROR,
492                      g_io_error_from_errno (errsv),
493                      _("Error opening file '%s': %s"),
494                      filename, g_strerror (errsv));
495       return NULL;
496     }
497   
498   stream = g_object_new (G_TYPE_LOCAL_FILE_OUTPUT_STREAM, NULL);
499   stream->priv->fd = fd;
500   
501   return G_FILE_OUTPUT_STREAM (stream);
502 }
503
504 static char *
505 create_backup_filename (const char *filename)
506 {
507   return g_strconcat (filename, BACKUP_EXTENSION, NULL);
508 }
509
510 #define BUFSIZE 8192 /* size of normal write buffer */
511
512 static gboolean
513 copy_file_data (gint     sfd,
514                 gint     dfd,
515                 GError **error)
516 {
517   gboolean ret = TRUE;
518   gpointer buffer;
519   const gchar *write_buffer;
520   gssize bytes_read;
521   gssize bytes_to_write;
522   gssize bytes_written;
523   
524   buffer = g_malloc (BUFSIZE);
525   
526   do
527     {
528       bytes_read = read (sfd, buffer, BUFSIZE);
529       if (bytes_read == -1)
530         {
531           if (errno == EINTR)
532             continue;
533           
534           g_set_error (error, G_IO_ERROR,
535                        g_io_error_from_errno (errno),
536                        _("Error reading from file: %s"),
537                        g_strerror (errno));
538           ret = FALSE;
539           break;
540         }
541       
542       bytes_to_write = bytes_read;
543       write_buffer = buffer;
544       
545       do
546         {
547           bytes_written = write (dfd, write_buffer, bytes_to_write);
548           if (bytes_written == -1)
549             {
550               if (errno == EINTR)
551                 continue;
552               
553               g_set_error (error, G_IO_ERROR,
554                            g_io_error_from_errno (errno),
555                            _("Error writing to file: %s"),
556                            g_strerror (errno));
557               ret = FALSE;
558               break;
559             }
560           
561           bytes_to_write -= bytes_written;
562           write_buffer += bytes_written;
563         }
564       while (bytes_to_write > 0);
565       
566     } while ((bytes_read != 0) && (ret == TRUE));
567
568   g_free (buffer);
569   
570   return ret;
571 }
572
573 static int
574 handle_overwrite_open (const char    *filename,
575                        const char    *etag,
576                        gboolean       create_backup,
577                        char         **temp_filename,
578                        GCancellable  *cancellable,
579                        GError       **error)
580 {
581   int fd = -1;
582   struct stat original_stat;
583   char *current_etag;
584   gboolean is_symlink;
585   int open_flags;
586
587   /* We only need read access to the original file if we are creating a backup.
588    * We also add O_CREATE to avoid a race if the file was just removed */
589   if (create_backup)
590     open_flags = O_RDWR | O_CREAT;
591   else
592     open_flags = O_WRONLY | O_CREAT;
593   
594   /* Some systems have O_NOFOLLOW, which lets us avoid some races
595    * when finding out if the file we opened was a symlink */
596 #ifdef O_NOFOLLOW
597   is_symlink = FALSE;
598   fd = g_open (filename, open_flags | O_NOFOLLOW, 0666);
599   if (fd == -1 && errno == ELOOP)
600     {
601       /* Could be a symlink, or it could be a regular ELOOP error,
602        * but then the next open will fail too. */
603       is_symlink = TRUE;
604       fd = g_open (filename, open_flags, 0666);
605     }
606 #else
607   fd = g_open (filename, open_flags, 0666);
608   /* This is racy, but we do it as soon as possible to minimize the race */
609   is_symlink = g_file_test (filename, G_FILE_TEST_IS_SYMLINK);
610 #endif
611     
612   if (fd == -1)
613     {
614       g_set_error (error, G_IO_ERROR,
615                    g_io_error_from_errno (errno),
616                    _("Error opening file '%s': %s"),
617                    filename, g_strerror (errno));
618       return -1;
619     }
620   
621   if (fstat (fd, &original_stat) != 0) 
622     {
623       g_set_error (error, G_IO_ERROR,
624                    g_io_error_from_errno (errno),
625                    _("Error stating file '%s': %s"),
626                    filename, g_strerror (errno));
627       goto err_out;
628     }
629   
630   /* not a regular file */
631   if (!S_ISREG (original_stat.st_mode))
632     {
633       if (S_ISDIR (original_stat.st_mode))
634         g_set_error (error,
635                      G_IO_ERROR,
636                      G_IO_ERROR_IS_DIRECTORY,
637                      _("Target file is a directory"));
638       else
639         g_set_error (error,
640                      G_IO_ERROR,
641                      G_IO_ERROR_NOT_REGULAR_FILE,
642                      _("Target file is not a regular file"));
643       goto err_out;
644     }
645   
646   if (etag != NULL)
647     {
648       current_etag = _g_local_file_info_create_etag (&original_stat);
649       if (strcmp (etag, current_etag) != 0)
650         {
651           g_set_error (error,
652                        G_IO_ERROR,
653                        G_IO_ERROR_WRONG_ETAG,
654                        _("The file was externally modified"));
655           g_free (current_etag);
656           goto err_out;
657         }
658       g_free (current_etag);
659     }
660
661   /* We use two backup strategies.
662    * The first one (which is faster) consist in saving to a
663    * tmp file then rename the original file to the backup and the
664    * tmp file to the original name. This is fast but doesn't work
665    * when the file is a link (hard or symbolic) or when we can't
666    * write to the current dir or can't set the permissions on the
667    * new file. 
668    * The second strategy consist simply in copying the old file
669    * to a backup file and rewrite the contents of the file.
670    */
671   
672   if (!(original_stat.st_nlink > 1) && !is_symlink)
673     {
674       char *dirname, *tmp_filename;
675       int tmpfd;
676       
677       dirname = g_path_get_dirname (filename);
678       tmp_filename = g_build_filename (dirname, ".goutputstream-XXXXXX", NULL);
679       g_free (dirname);
680
681       tmpfd = g_mkstemp (tmp_filename);
682       if (tmpfd == -1)
683         {
684           g_free (tmp_filename);
685           goto fallback_strategy;
686         }
687       
688       /* try to keep permissions */
689
690       if (
691 #ifdef F_CHOWN
692           fchown (tmpfd, original_stat.st_uid, original_stat.st_gid) == -1 ||
693 #endif
694 #ifdef F_CHMOD
695           fchmod (tmpfd, original_stat.st_mode) == -1 ||
696 #endif
697           0
698           )
699         {
700           struct stat tmp_statbuf;
701           
702           /* Check that we really needed to change something */
703           if (fstat (tmpfd, &tmp_statbuf) != 0 ||
704               original_stat.st_uid != tmp_statbuf.st_uid ||
705               original_stat.st_gid != tmp_statbuf.st_gid ||
706               original_stat.st_mode != tmp_statbuf.st_mode)
707             {
708               close (tmpfd);
709               unlink (tmp_filename);
710               g_free (tmp_filename);
711               goto fallback_strategy;
712             }
713         }
714
715       close (fd);
716       *temp_filename = tmp_filename;
717       return tmpfd;
718     }
719
720  fallback_strategy:
721
722   if (create_backup)
723     {
724       struct stat tmp_statbuf;      
725       char *backup_filename;
726       int bfd;
727       
728       backup_filename = create_backup_filename (filename);
729
730       if (unlink (backup_filename) == -1 && errno != ENOENT)
731         {
732           g_set_error (error,
733                        G_IO_ERROR,
734                        G_IO_ERROR_CANT_CREATE_BACKUP,
735                        _("Backup file creation failed"));
736           g_free (backup_filename);
737           goto err_out;
738         }
739
740       bfd = g_open (backup_filename,
741                     O_WRONLY | O_CREAT | O_EXCL,
742                     original_stat.st_mode & 0777);
743
744       if (bfd == -1)
745         {
746           g_set_error (error,
747                        G_IO_ERROR,
748                        G_IO_ERROR_CANT_CREATE_BACKUP,
749                        _("Backup file creation failed"));
750           g_free (backup_filename);
751           goto err_out;
752         }
753
754       /* If needed, Try to set the group of the backup same as the
755        * original file. If this fails, set the protection
756        * bits for the group same as the protection bits for
757        * others. */
758 #ifdef HAVE_FCHOWN
759       if (fstat (bfd, &tmp_statbuf) != 0)
760         {
761           g_set_error (error,
762                        G_IO_ERROR,
763                        G_IO_ERROR_CANT_CREATE_BACKUP,
764                        _("Backup file creation failed"));
765           unlink (backup_filename);
766           g_free (backup_filename);
767           goto err_out;
768         }
769       
770       if ((original_stat.st_gid != tmp_statbuf.st_gid)  &&
771           fchown (bfd, (uid_t) -1, original_stat.st_gid) != 0)
772         {
773           if (fchmod (bfd,
774                       (original_stat.st_mode & 0707) |
775                       ((original_stat.st_mode & 07) << 3)) != 0)
776             {
777               g_set_error (error,
778                            G_IO_ERROR,
779                            G_IO_ERROR_CANT_CREATE_BACKUP,
780                            _("Backup file creation failed"));
781               unlink (backup_filename);
782               close (bfd);
783               g_free (backup_filename);
784               goto err_out;
785             }
786         }
787 #endif
788
789       if (!copy_file_data (fd, bfd, NULL))
790         {
791           g_set_error (error,
792                        G_IO_ERROR,
793                        G_IO_ERROR_CANT_CREATE_BACKUP,
794                        _("Backup file creation failed"));
795           unlink (backup_filename);
796           close (bfd);
797           g_free (backup_filename);
798           
799           goto err_out;
800         }
801       
802       close (bfd);
803       g_free (backup_filename);
804
805       /* Seek back to the start of the file after the backup copy */
806       if (lseek (fd, 0, SEEK_SET) == -1)
807         {
808           g_set_error (error, G_IO_ERROR,
809                        g_io_error_from_errno (errno),
810                        _("Error seeking in file: %s"),
811                        g_strerror (errno));
812           goto err_out;
813         }
814     }
815
816   /* Truncate the file at the start */
817 #ifdef G_OS_WIN32
818   if (g_win32_ftruncate (fd, 0) == -1)
819 #else
820   if (ftruncate (fd, 0) == -1)
821 #endif
822     {
823       g_set_error (error, G_IO_ERROR,
824                    g_io_error_from_errno (errno),
825                    _("Error truncating file: %s"),
826                    g_strerror (errno));
827       goto err_out;
828     }
829     
830   return fd;
831
832  err_out:
833   close (fd);
834   return -1;
835 }
836
837 GFileOutputStream *
838 _g_local_file_output_stream_replace (const char        *filename,
839                                      const char        *etag,
840                                      gboolean           create_backup,
841                                      GFileCreateFlags   flags,
842                                      GCancellable      *cancellable,
843                                      GError           **error)
844 {
845   GLocalFileOutputStream *stream;
846   int mode;
847   int fd;
848   char *temp_file;
849
850   if (g_cancellable_set_error_if_cancelled (cancellable, error))
851     return NULL;
852
853   temp_file = NULL;
854
855   if (flags & G_FILE_CREATE_PRIVATE)
856     mode = 0600;
857   else
858     mode = 0666;
859
860   /* If the file doesn't exist, create it */
861   fd = g_open (filename, O_CREAT | O_EXCL | O_WRONLY, mode);
862
863   if (fd == -1 && errno == EEXIST)
864     {
865       /* The file already exists */
866       fd = handle_overwrite_open (filename, etag, create_backup, &temp_file,
867                                   cancellable, error);
868       if (fd == -1)
869         return NULL;
870     }
871   else if (fd == -1)
872     {
873       int errsv = errno;
874
875       if (errsv == EINVAL)
876         /* This must be an invalid filename, on e.g. FAT */
877         g_set_error (error, G_IO_ERROR,
878                      G_IO_ERROR_INVALID_FILENAME,
879                      _("Invalid filename"));
880       else
881         g_set_error (error, G_IO_ERROR,
882                      g_io_error_from_errno (errsv),
883                      _("Error opening file '%s': %s"),
884                      filename, g_strerror (errsv));
885       return NULL;
886     }
887   
888  
889   stream = g_object_new (G_TYPE_LOCAL_FILE_OUTPUT_STREAM, NULL);
890   stream->priv->fd = fd;
891   stream->priv->tmp_filename = temp_file;
892   if (create_backup)
893     stream->priv->backup_filename = create_backup_filename (filename);
894   stream->priv->original_filename =  g_strdup (filename);
895   
896   return G_FILE_OUTPUT_STREAM (stream);
897 }