Fix pg_file_write() error handling.
authorNoah Misch
Sun, 12 Mar 2017 23:35:31 +0000 (19:35 -0400)
committerNoah Misch
Sun, 12 Mar 2017 23:36:15 +0000 (19:36 -0400)
Detect fclose() failures; given "ln -s /dev/full $PGDATA/devfull",
"pg_file_write('devfull', 'x', true)" now fails as it should.  Don't
leak a stream when fwrite() fails.  Remove a born-ineffective test that
aimed to skip zero-length writes.  Back-patch to 9.2 (all supported
versions).

contrib/adminpack/adminpack.c

index ded89c60590322f69281675cdbb506576ca9ac97..23751f380f4e17c48233ac48fcab425fda93fb8c 100644 (file)
@@ -141,10 +141,10 @@ pg_file_write(PG_FUNCTION_ARGS)
                    (ERRCODE_DUPLICATE_FILE,
                     errmsg("file \"%s\" exists", filename)));
 
-       f = fopen(filename, "wb");
+       f = AllocateFile(filename, "wb");
    }
    else
-       f = fopen(filename, "ab");
+       f = AllocateFile(filename, "ab");
 
    if (!f)
        ereport(ERROR,
@@ -152,16 +152,11 @@ pg_file_write(PG_FUNCTION_ARGS)
                 errmsg("could not open file \"%s\" for writing: %m",
                        filename)));
 
-   if (VARSIZE(data) != 0)
-   {
-       count = fwrite(VARDATA(data), 1, VARSIZE(data) - VARHDRSZ, f);
-
-       if (count != VARSIZE(data) - VARHDRSZ)
-           ereport(ERROR,
-                   (errcode_for_file_access(),
-                    errmsg("could not write file \"%s\": %m", filename)));
-   }
-   fclose(f);
+   count = fwrite(VARDATA(data), 1, VARSIZE(data) - VARHDRSZ, f);
+   if (count != VARSIZE(data) - VARHDRSZ || FreeFile(f))
+       ereport(ERROR,
+               (errcode_for_file_access(),
+                errmsg("could not write file \"%s\": %m", filename)));
 
    PG_RETURN_INT64(count);
 }