From b6b6b79bfd14c3ec9c19606c3da59600301dd195 Mon Sep 17 00:00:00 2001 From: Robert Hague Date: Sun, 15 Jan 2023 14:49:01 +0000 Subject: [PATCH 1/2] Add check for hasHeaderRecord in CsvWriter.WriteHeader(IEnumerable). Also write the header of the first non-null record rather than the last fixes #2092 --- src/CsvHelper/CsvWriter.cs | 10 +++++--- tests/CsvHelper.Tests/CsvWriterTests.cs | 32 +++++++++++++++++++++++++ 2 files changed, 39 insertions(+), 3 deletions(-) diff --git a/src/CsvHelper/CsvWriter.cs b/src/CsvHelper/CsvWriter.cs index 5cf50d6ba..770a6a7a0 100644 --- a/src/CsvHelper/CsvWriter.cs +++ b/src/CsvHelper/CsvWriter.cs @@ -803,16 +803,20 @@ private bool WriteHeader(IEnumerable records) private bool WriteHeader(IEnumerable records) { - object? record = null; + if (!hasHeaderRecord || hasHeaderBeenWritten) + { + return false; + } + foreach (var r in records) { if (r != null) { - record = r; + return WriteHeader(r); } } - return WriteHeader(record); + return false; } private bool WriteHeader(object? record) diff --git a/tests/CsvHelper.Tests/CsvWriterTests.cs b/tests/CsvHelper.Tests/CsvWriterTests.cs index 2309ef7d1..3b5193aa7 100644 --- a/tests/CsvHelper.Tests/CsvWriterTests.cs +++ b/tests/CsvHelper.Tests/CsvWriterTests.cs @@ -13,6 +13,7 @@ using System.Dynamic; using Xunit; using System.Threading; +using System.Collections; namespace CsvHelper.Tests { @@ -835,6 +836,37 @@ public void WriteInternalConstructorClassTest() } } + [Theory] + [InlineData(false)] + [InlineData(true)] + public void WriteRecords_NonGeneric_HasHeaderRecord(bool hasHeaderRecord) + { + var confg = new CsvConfiguration(CultureInfo.InvariantCulture) + { + HasHeaderRecord = hasHeaderRecord + }; + + using var stringWriter = new StringWriter(); + using var csvWriter = new CsvWriter(stringWriter, confg); + + IEnumerable records = new object[] { + new TestSinglePropertyRecord { Name = "test" }, + new TestRecord { IntColumn = 4 } + }; + + csvWriter.WriteRecords(records); + var csv = stringWriter.ToString(); + + if (hasHeaderRecord) + { + Assert.Equal("Name\r\ntest\r\n4,,,,\r\n", csv); + } + else + { + Assert.Equal("test\r\n4,,,,\r\n", csv); + } + } + private class GetOnly { internal GetOnly(string someParam) From bf460aceed063fc999f4d9ec524707122cf19bf5 Mon Sep 17 00:00:00 2001 From: Josh Close Date: Fri, 26 Jan 2024 09:39:37 -0600 Subject: [PATCH 2/2] Cleanup. --- src/CsvHelper/CsvWriter.cs | 20 +++++++++++--------- tests/CsvHelper.Tests/CsvWriterTests.cs | 15 +++++++-------- 2 files changed, 18 insertions(+), 17 deletions(-) diff --git a/src/CsvHelper/CsvWriter.cs b/src/CsvHelper/CsvWriter.cs index 3d9d8dde8..768049d1f 100644 --- a/src/CsvHelper/CsvWriter.cs +++ b/src/CsvHelper/CsvWriter.cs @@ -2,23 +2,22 @@ // This file is a part of CsvHelper and is dual licensed under MS-PL and Apache 2.0. // See LICENSE.txt for details or visit http://www.opensource.org/licenses/ms-pl.html for MS-PL and http://opensource.org/licenses/Apache-2.0 for Apache 2.0. // https://github.com/JoshClose/CsvHelper +using CsvHelper.Configuration; +using CsvHelper.Expressions; +using CsvHelper.TypeConversion; using System; +using System.Buffers; using System.Collections; using System.Collections.Generic; +using System.Dynamic; +using System.Globalization; using System.IO; -using System.Reflection; -using CsvHelper.Configuration; -using CsvHelper.TypeConversion; using System.Linq; using System.Linq.Expressions; -using System.Dynamic; -using System.Threading.Tasks; -using CsvHelper.Expressions; -using System.Globalization; +using System.Reflection; using System.Runtime.CompilerServices; -using System.Text; -using System.Buffers; using System.Threading; +using System.Threading.Tasks; #pragma warning disable 649 #pragma warning disable 169 @@ -795,6 +794,7 @@ private bool WriteHeader(IEnumerable records) if (!isPrimitive && recordType != typeof(object)) { WriteHeader(recordType); + return hasHeaderBeenWritten; } @@ -829,6 +829,7 @@ private bool WriteHeader(object record) if (record is IDynamicMetaObjectProvider dynamicObject) { WriteDynamicHeader(dynamicObject); + return true; } @@ -837,6 +838,7 @@ private bool WriteHeader(object record) if (!isPrimitive) { WriteHeader(recordType); + return true; } diff --git a/tests/CsvHelper.Tests/CsvWriterTests.cs b/tests/CsvHelper.Tests/CsvWriterTests.cs index d0a5913e9..738e13827 100644 --- a/tests/CsvHelper.Tests/CsvWriterTests.cs +++ b/tests/CsvHelper.Tests/CsvWriterTests.cs @@ -2,22 +2,21 @@ // This file is a part of CsvHelper and is dual licensed under MS-PL and Apache 2.0. // See LICENSE.txt for details or visit http://www.opensource.org/licenses/ms-pl.html for MS-PL and http://opensource.org/licenses/Apache-2.0 for Apache 2.0. // https://github.com/JoshClose/CsvHelper +using CsvHelper.Configuration; +using CsvHelper.TypeConversion; using System; +using System.Collections; using System.Collections.Generic; +using System.Dynamic; using System.Globalization; using System.IO; -using System.Text; -using CsvHelper.Configuration; -using CsvHelper.TypeConversion; -using Int32Converter = CsvHelper.TypeConversion.Int32Converter; -using System.Dynamic; -using Xunit; using System.Threading; -using System.Collections; +using Xunit; +using Int32Converter = CsvHelper.TypeConversion.Int32Converter; namespace CsvHelper.Tests { - + public class CsvWriterTests { public CsvWriterTests()