Generate a Flat file with ANSI encoding using Logic App

Handling non UTF8 encoding in Logic App

Recently there was a requirement to generate a flat file in Azure Logic App and deliver to Azure File Share with ANSI encoding as targeting application could only process ANSI encoding file. Much of cloud services assume that a text payload will be some form of UTF (Unicode) encoding. Azure Logic App Assumes it is UTF-8. Such that when your text payload is in a different encoding, such as a page code based encoding, the non-basic Latin characters gets mangled. This is particularly common with Flat Files because they integrate with ancient systems that often were not written with Unicode support.

My approach to solving the problem was to create an Azure Function App that converts the encoding from UTF-8 to windows-1252 (or to any other encoding) and then stores the file content in Azure File storage.

Azure Integration Logic APP ANSI Encoding using Azure Function App

This seems to be an easy fix but the main problem was that Azure Logic App did not like the output from Azure Function App and threw an exception as shown below

BadRequest. Http request failed as the content was not valid: ‘Unable to translate bytes [E4] at index 83 from specified code page to Unicode.’.

Azure Integration Logic APP ANSI Encoding using Azure Function App failure exception

The solution would be to use Base 64 encoding. Base 64 encoding ensures that none of the services in Azure integration going to assume a UTF encoding. Once you convert any non-UTF flat file such (as windows-1252) to UTF-8, then base 64 decodes it safely and process it with flat-file decode.

Azure Function App to change the encoding

The following azure function app can be used to convert the encoding of the text in base64 encoding

using System;
using System.Net;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;
using Microsoft.Azure.WebJobs;
using Microsoft.Azure.WebJobs.Host;
using Newtonsoft.Json;

namespace PnJ.FunctionApp.ConvertEncoding
{
    public static class ChangeBase64Encoding
    {
        [FunctionName("ChangeBase64Encoding")]
        public static async Task<object> Run([HttpTrigger(WebHookType = "genericJson")]HttpRequestMessage req, TraceWriter log)
        {
            log.Info($"Change base 64 Encoding function App was triggered");

            Encoding inputEncoding = null;

            string jsonContent = await req.Content.ReadAsStringAsync();
            dynamic data = JsonConvert.DeserializeObject(jsonContent);

            if (data == null || data.text == null || data.encodingInput == null || data.encodingOutput == null)
            {
                return req.CreateResponse(HttpStatusCode.BadRequest, new
                {
                    error = "Please pass text/encodingOutput properties in the input Json object."
                });
            }

            try
            {
                string encodingInput = data.encodingInput;
                inputEncoding = Encoding.GetEncoding(name: encodingInput);
            }
            catch (ArgumentException)
            {
                return req.CreateResponse(HttpStatusCode.BadRequest, new
                {
                    error = "Input char set value '" + data.encodingInput + "' is not supported. Supported value are listed at https://msdn.microsoft.com/en-us/library/system.text.encoding(v=vs.110).aspx."
                });
            }

            Encoding encodingOutput;
            try
            {
                string outputEncoding = data.encodingOutput;
                encodingOutput = Encoding.GetEncoding(outputEncoding);
            }
            catch (ArgumentException)
            {
                return req.CreateResponse(HttpStatusCode.BadRequest, new
                {
                    error = "Output char set value '" + data.encodingOutput + "' is not supported. Supported value are listed at https://msdn.microsoft.com/en-us/library/system.text.encoding(v=vs.110).aspx."
                });
            }

            string input = data.text;
            var outputBytes = Encoding.Convert(srcEncoding: inputEncoding, dstEncoding: encodingOutput, bytes: Convert.FromBase64String(input));

            var response = req.CreateResponse(HttpStatusCode.OK);
            response.Content = new StringContent(content: JsonConvert.SerializeObject(new
            {
                text = Convert.ToBase64String(outputBytes)
            }).ToString(), encoding: encodingOutput, mediaType: "application/json");

            return response;
        }
    }
}

The function app receives the following input and encodes to the desired format, sends it back.

{
  "encodingInput": "utf-8",
  "encodingOutput": "windows-1252",
  "text": "U0hQMDAwMDExMzZ8VHN1YmFraSBFdXJvcGUgQi5WLnxBdmVudHVyaWpufDMzMTYgTEJ8RG9yZHJlY2h0fE5MfDE4NDMwOHxSdXRoZW5iZXJnIExhbmR0ZWNobmlrfENhcmwtQm9yZ3dhcmQtU3RyIDF8R8O8dGVyc2xvaHwzMzMzNXxERXwyMDIwMDQyOXw0OE58U3xCb3ggM3xDaGFpbiBhbmQgcGFydHN8M3wwLjAyfA=="
}

If the above approach doesn’t work, then please use the following approach, which gave me the desired results.

using System;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Net.Http.Headers;
using System.Text;
using System.Threading.Tasks;
using Microsoft.Azure.WebJobs;
using Microsoft.Azure.WebJobs.Extensions.Http;
using Microsoft.Azure.WebJobs.Host;
using Newtonsoft.Json;

namespace PnJ.FunctionApp.ConvertEncoding
{
    public static class ChangeEncoding
    {
        [FunctionName("ChangeEncoding")]
        public static async Task<HttpResponseMessage> Run([HttpTrigger(AuthorizationLevel.Function, "get", "post", Route = null)]HttpRequestMessage req, TraceWriter log)
        {
            log.Info($"Change Encoding function App was triggered was triggered!");

            Encoding inputEncoding = null;

            string jsonContent = await req.Content.ReadAsStringAsync();
            dynamic data = JsonConvert.DeserializeObject(jsonContent);

            if (data == null || data.text == null || data.encodingInput == null || data.encodingOutput == null)
            {
                return req.CreateResponse(HttpStatusCode.BadRequest, new
                {
                    error = "Please pass text/encodingOutput properties in the input Json object."
                });
            }
            try
            {
                string encodingInput = data.encodingInput;
                inputEncoding = Encoding.GetEncoding(name: encodingInput);
            }
            catch (ArgumentException)
            {
                return req.CreateResponse(HttpStatusCode.BadRequest, new
                {
                    error = "Input char set value '" + data.encodingInput + "' is not supported. Supported value are listed at https://msdn.microsoft.com/en-us/library/system.text.encoding(v=vs.110).aspx."
                });
            }

            Encoding encodingOutput = null;
            try
            {
                string outputEncoding = data.encodingOutput;
                encodingOutput = Encoding.GetEncoding(outputEncoding);
            }
            catch (ArgumentException)
            {
                return req.CreateResponse(HttpStatusCode.BadRequest, new
                {
                    error = "Output char set value '" + data.encodingOutput + "' is not supported. Supported value are listed at https://msdn.microsoft.com/en-us/library/system.text.encoding(v=vs.110).aspx."
                });
            }

            string input = data.text;
            var outputBytes = Encoding.Convert(srcEncoding: inputEncoding, dstEncoding: encodingOutput,  inputEncoding.GetBytes(input));
            var response = req.CreateResponse(HttpStatusCode.OK);
            MemoryStream ms = new MemoryStream(outputBytes);
            response.Content = new StreamContent(ms);
            response.Content.Headers.ContentType = new MediaTypeHeaderValue("application/octet-stream");
            return response;
        }
    }
}
{
  "encodingInput": "utf-8",
  "encodingOutput": "windows-1252",
  "text": "U0hQMDAwMDExMzZ8VHN1YmFraSBFdXJvcGUgQi5WLnxBdmVudHVyaWpufDMzMTYgTEJ8RG9yZHJlY2h0fE5MfDE4NDMwOHxSdXRoZW5iZXJnIExhbmR0ZWNobmlrfENhcmwtQm9yZ3dhcmQtU3RyIDF8R8O8dGVyc2xvaHwzMzMzNXxERXwyMDIwMDQyOXw0OE58U3xCb3ggM3xDaGFpbiBhbmQgcGFydHN8M3wwLjAyfA=="
}

I created following Azure to Logic App to test the out come. The outcome from the second approach gave me better results.

Azure Integration Logic APP ANSI Encoding using Azure Function App

    Published by Poojith Jain

    Poojith Jain is an Azure Architect with good experience with software design and development. He has a thorough knowledge of Azure Integration and he is passionate about solving complex and challenging problems in the field of Azure

    One thought on “Generate a Flat file with ANSI encoding using Logic App

    Leave a Reply

    Fill in your details below or click an icon to log in:

    WordPress.com Logo

    You are commenting using your WordPress.com account. Log Out /  Change )

    Twitter picture

    You are commenting using your Twitter account. Log Out /  Change )

    Facebook photo

    You are commenting using your Facebook account. Log Out /  Change )

    Connecting to %s

    This site uses Akismet to reduce spam. Learn how your comment data is processed.

    %d bloggers like this: