Partilhar via


Missing Byte Order Mark in Flat file generated with Flat File Connection Manager in SSIS

 

I have recently worked on a problem where we come across a situation such as the following:

You are working on a solution using SQL Server Integration Services(Visual Studio 2005).
You are pulling data from your database and trying to place the results into a flat file (.CSV) in UTF-8 format. The solution exports the data perfectly and keeps the special characters in the file because you have used 65001 as the code page.

clip_image002

However, the text file when you open it or try to load it to another process, it says the file is ANSI instead of UTF-8. If you open the file in notepad and do a SAVE AS and change the encode to UTF-8 and then your external process works but this is a tedious manual work.

What I have found that when you specify the Code Page property of the Flat file connection manager, it do generates a UTF-8 file. However, it generates a version of the UTF-8 file which misses something we call as Byte Order Mark.

So if you have a CSV file containing the character AA, the BOM for UTF8 will be 0xef, 0xbb and 0xbf. Even though the file has no BOM, it’s still UTF8.

clip_image002[4]

Unfortunately, in some old legacy systems, the applications search for the BOM to determine the type of the file. It appears that your process is also doing the same.

To workaround the problem you can use the following piece of code in your script task which can be ran after the export process.

 static void Main(string[] args)
       {
           string pattern = "*.csv";
           string[] files = Directory.GetFiles(@".\", pattern, SearchOption.AllDirectories);
           FileCodePageConverter converter = new FileCodePageConverter();
           converter.SetCulture("en-US");
           foreach (string file in files)
           {
               converter.Convert(file, file, "Windows-1252"); // Convert from code page Windows-1250 to UTF-8  
           }  
       }

class FileCodePageConverter

  {

      public void Convert(string path, string path2, string codepage)

      {

          byte[] buffer = File.ReadAllBytes(path);

          if (buffer[0] != 0xef && buffer[0] != 0xbb)

          {

              byte[] buffer2 = Encoding.Convert(Encoding.GetEncoding(codepage), Encoding.UTF8, buffer);

              byte[] utf8 = new byte[] { 0xef, 0xbb, 0xbf };

              FileStream fs = File.Create(path2);

              fs.Write(utf8, 0, utf8.Length);

              fs.Write(buffer2, 0, buffer2.Length);

              fs.Close();

          }

      }

      public void SetCulture(string name)

      {

          Thread.CurrentThread.CurrentCulture = new CultureInfo(name);

          Thread.CurrentThread.CurrentUICulture = new CultureInfo(name);

      }

  }

when you will run the package you will find that all the CSVs in the designated folder will be converted into a UTF8 format which contains the byte order mark.

This way your external process will be able to work with the exported CSV files.

I hope this helps!!

~Jay

Comments

  • Anonymous
    January 11, 2012
    Thanks Jay, this is a great explanation and was very helpful. We were finding some files were being designated as "ANSI as UTF-8" and I think this will help us.

  • Anonymous
    January 28, 2013
    Thanks, very helpful snippet. Works fine.

  • Anonymous
    August 19, 2014
    Thanks for the above code...it helped me lot... just add below namespaces..... using System.IO; using System.Text; using System.Threading; using System.Globalization; if you are looking only for particular folder...send that variable to script task and use below one..           string sPath;           sPath=Dts.Variables["User::v_ExtractPath"].Value.ToString();           string pattern = "*.txt";           string[] files = Directory.GetFiles(sPath);