FixedWidthFileUtils 1.3.1

dotnet add package FixedWidthFileUtils --version 1.3.1
NuGet\Install-Package FixedWidthFileUtils -Version 1.3.1
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="FixedWidthFileUtils" Version="1.3.1" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add FixedWidthFileUtils --version 1.3.1
#r "nuget: FixedWidthFileUtils, 1.3.1"
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install FixedWidthFileUtils as a Cake Addin
#addin nuget:?package=FixedWidthFileUtils&version=1.3.1

// Install FixedWidthFileUtils as a Cake Tool
#tool nuget:?package=FixedWidthFileUtils&version=1.3.1

About

Tool for serializing objects to fixed width files, or deserializing fixed width files to objects.

Current largest file test contained 1,179,648 records in the Wells Fargo format, and took ~10 seconds to deserialize.

Some rules to keep in mind:

  • Position of the FixedField defines the order of the value rather than the start index of the substring - EG: 00001 Michael: though the name field starts at character index 5, its position is the second element, position would be 1 (elements use a 0 index position)
  • Value types can not be directly serialized / deserialized (ie: FixedWidthSerializer.Deserialize<int>("00001"); is not valid)
  • All properties must be decorated with FixedField, with a minimum of the position provided
  • An object can contains value types, or complex types, not both
  • Each object containing value types must be its own line

Examples

Serialization

public class Person
{
    //field at position 0 is 10 long, pad with spaces, and align the text to the left side of the field (Right is default)
    //NOTE: Field position defines the order of fields, rather than the index that the substring of the field starts at
    [FixedField(0, 10, ' ', FixedFieldAlignment.Left, FixedFieldOverflowMode.Truncate)] //align field to the left, and just truncate if their name is too long - their ID is the import bit!
    public string FirstName { get; set; }

    [FixedField(1, 10, ' ', FixedFieldAlignment.Left, FixedFieldOverflowMode.Truncate)]
    public string LastName { get; set; }

    [FixedField(2, 10)] //default is right aligned and padding with 0's
    public int ID { get; set; }
}

Person p = new Person
{
    FirstName = "Mike",
    LastName = "Jones",
    ID = 1234
};

string result = FixedWidthSerializer.Serialize(Person);
//Mike      Jones     0000001234

Deserialization

//sample file contains:
/*
Mike      Jones     0000001234
Bill      Jones     0000001235
*/
List<Person> people = null;
using (FileStream fs = new FileStream(sampleFile))
    people = FixedWidthSerializer.Deserialize<List<Person>>(fs);
if (people != null)
{
    foreach (var person in people)
        Console.WriteLine($"{person.LastName}, {person.FirstName} has an ID of {person.ID}");
}

A bit more complex..

Mixing value fields and complex (class) fields is not supported - you must nest instead. For example, Wells Fargo requires check records to be grouped by account number and include a trailer. You can achieve this by building your models like so:

public class CheckGroup
{
    [FixedField(0)]
    public CheckRecord[] Records { get; set; }

    [FixedField(1)]
    public CheckGroupTrailer Trailer { get; set; }
}
public class CheckRecord
{
    [FixedField(0, 10)]
    public long CheckSerial { get; set; }

    [FixedField(1, 6)]
    [FixedFieldSerializer(typeof(WellsFargoDateSerializer))]
    public DateTime IssueDate { get; set; }

    [FixedField(2, 15)]
    public long AccountNumber { get; set; }

    [FixedField(3, 3)]
    public int TransactionCode => 320;

    [FixedField(4, 10)]
    [FixedFieldSerializer(typeof(DecimalToPenniesSerializer))]
    public decimal Amount { get; set; }

    [FixedField(5, 41, ' ', FixedFieldAlignment.Left)]
    public string Payee { get; set; }
}
[FixedObjectPattern("^&")]
public class CheckGroupTrailer
{
    [FixedField(0, 15, ' ', FixedFieldAlignment.Left)]
    public string Start => "&";

    [FixedField(1, 5)]
    public int RecordCount { get; set; }

    [FixedField(2, 3, ' ')]
    [FixedField(4, 52, ' ')] //a property can be used multiple times in an object's string. Mostly useful for placeholder properties
    public string Spacer => string.Empty;

    [FixedField(3, 10)]
    [FixedFieldSerializer(typeof(DecimalToPenniesSerializer))]
    public decimal TotalAmount { get; set; }
}

Once you build your models like this, you could then Deserialize the following input:

000000604201292000000XXXXXXXXXX3200000016809WELLS FARGO BANK N.A.
&              00001   0000016809
000001998201292000000XXXXXXXXXX3200000340683SGS NORTH AMERICA
&              00001   0000340683
000003667501292000000XXXXXXXXXX3200000382792SAMS, LARKIN & HUFF
00000366760129200000XXXXXXXXXXX3200000352979HARKLEROAD & ASSOC., INC.
000003667701292000000XXXXXXXXXX3200000022175WELLS FARGO BANK N.A.
&              00003   0000757946
000002372401292000000XXXXXXXXXX3200000243300SAMS, LARKIN & HUFF
000002372501292000000XXXXXXXXXX3200002900000C&M EQUIPMENT
000002372601292000000XXXXXXXXXX3200000079753CONSOLIDATED ELECTRICAL DIST., INC.
000002372701292000000XXXXXXXXXX3200000012500EPIC PARTNERS INSURANCE CENTER
000002372801292000000XXXXXXXXXX3200000091500ROLL-A-SHADE, INC.
000002372901292000000XXXXXXXXXX3200001846417SUNSHINE ELECTRONIC DISPLAY
000002373001292000000XXXXXXXXXX3200000021267WELLS FARGO BANK N.A.
&              00007   0005194737

With a simple call:

public static void Main()
{
    string inputString = "...";
    CheckGroup[] checkGroups = FixedWidthSerializer.Deserialize<CheckGroup[]>(inputString);

    //and then serialize it again...

    string fixedWidthFileContent = FixedWidthSerializer.Serialize(checkGroups);
}

Custom Serializer

public class WellsFargoDateSerializer : FixedFieldSerializer<DateTime>
{
    public override DateTime Deserialize(string input) => DateTime.ParseExact(input, "MMddyy", CultureInfo.InvariantCulture);
    public override string Serialize(DateTime input) => input.ToString("MMddyy");
}

public class SampleObject
{
    [FixedField(0, 6)]
    [FixedFieldSerializer(typeof(WellsFargoDateSerializer))]
    public DateTime IssueDate { get; set; }
}

var sample = new SampleObject
{
    IssueDate = DateTime.Now
};

Console.WriteLine(FixedWidthSerializer.Serialize(sample)); //012920

Console.WriteLine(FixedWidthSerializer.Deserialize<SampleObject>("012920").IssueDate); //1/29/2020 12:00:00 AM

Object Pattern Indicators

Often times, all lines in a file are the same width. This is not an issue if you aren't dealing with nested objects with collections, but it can present some issues when that is the case. If you need to provide a regular expression indicator to differentiate object types, you can do so like so:

[FixedObjectPattern("^&")] //trailer always starts with &, and is only line type which does so.
public class Trailer
{
    ...
}
Product Compatible and additional computed target framework versions.
.NET Framework net35 is compatible.  net40 was computed.  net403 was computed.  net45 was computed.  net451 was computed.  net452 was computed.  net46 was computed.  net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

This package has no dependencies.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.3.1 504 10/5/2020
1.3.0 636 2/21/2020
1.2.0 565 2/18/2020
1.1.0 516 2/11/2020
1.0.0 599 2/11/2020

Fixed bug when parsing nested array structures.