You are viewing limited content. For full access, please sign in.

Question

Question

How to export multivalue token into csv file

asked on May 15, 2015

Hi everyone,

I have found a few articles here on Answers that have helped me built this script. When I run this script as test directly on the script editor and use test tokens, it works perfectly. However, when I run this through workflow and get the metadata from the entries, I get the following error:

 

Here's my code:

Imports System
Imports System.Collections.Generic
Imports System.ComponentModel
Imports System.Data
Imports System.Data.SqlClient
Imports System.Text

Namespace WorkflowActivity.Scripting.Script
   '''<summary>
    '''Provides one or more methods that can be run when the workflow scripting activity is performed.
    '''</summary>
    Public Class Script1
        Inherits ScriptClass90
        '''<summary>
        '''This method is run when the activity is performed.
        '''</summary>
        Protected Overrides Sub Execute()

    'Wrap the code in a Try/Catch to catch any errors...
    Try
       'Instantiate a new streamwriter...
        Dim fileName as String
        Dim numRows As Integer = GetTokenValue("RowCount")
        Dim regDate as Date = Date.Now()
        Dim strDate as String = regDate.ToString("ddMMMyyyy")
        fileName = TokenReplace("Weekly Report - From "+"%(DateTokenCalculator_1WeekAgo#""MM-dd-yyyy""#) To "+"%(Date#""MM-dd-yyyy""#)")
        My.Computer.FileSystem.CreateDirectory("D:\TestData\"+strDate+"\")
        Dim csvWriter As New System.IO.StreamWriter("D:\TestData\"+strDate+"\"+fileName+".csv")
        Dim headerRow As String 'Will hold the header row values
        Dim dataRow As String   'Will hold the data row values
        Dim csvData As New StringBuilder    'We will append all of the data to this string object

        'Here is the header row...
        headerRow = "Employee Name,GL Account,Rate,Job Name,Job#,Reg Hrs Worked,Over Time Hrs Worked,Deduction Amount,Deduction Code(s)"
        csvData.AppendLine(headerRow)

        'Here is the first data row...
        For Index As Integer = 1 To numRows
           dataRow = TokenReplace("%(EmployeeName#["+Index.ToString+"]#)"+","+"%(GLAccount#["+Index.ToString+"]#)"+",,"+"%(JobName#["+Index.ToString+"]#)"+","+"%(JobNo#["+Index.ToString+"]#)"+","+"%(RegHrsWorked#["+Index.ToString+"]#)"+",,,")
           csvData.AppendLine(dataRow)
           Next Index

        'Now write the CSV file and close the streamwriter...
        csvWriter.Write(csvData)
        csvWriter.Close

        'Cleanup...
        csvWriter = Nothing
        csvData = Nothing

    Catch ex As Exception
        'Send any error messages to workflow so they can be tracked...
        WorkflowApi.TrackError(ex.message)

    End Try

End Sub

    End Class
End Namespace

Any help will be greatly appreciated!

0 0

Answer

SELECTED ANSWER
replied on May 18, 2015 Show version history

Edgar - Once I imported the workflow and documents you provided I can see what you are trying to do.  There were several errors but once corrected I can see a created CSV with document data.

The primary issues were that the RowCount token that you were passing to the script did not actually contain a numeric value but a string value of an EmployeeName. That is what was throwing the error you first reported.

After looking at the full workflow I realized that you do not need to do any TokenReplace on the passed multi-value string fields as the tokens have already been resolved by the 'RetrieveFieldValues' activity!

So here is the modified script;

    Try
        Dim strDate as String = Date.Now().ToString("ddMMMyyyy")
        Dim folderName As String = "D:\TestData\" & strDate & "\"
        Dim fileName As String = TokenReplace("Weekly Report - From "+"%(DateTokenCalculator_1WeekAgo#""MM-dd-yyyy""#) To "+"%(Date#""MM-dd-yyyy""#)")
        
        'Create the folder...
        My.Computer.FileSystem.CreateDirectory(folderName)
        
        'Open the streamwriter...
        Dim csvWriter As New System.IO.StreamWriter(folderName & fileName & ".csv")
        Dim headerRow As String 'Will hold the header row values
        Dim csvData As New StringBuilder    'We will append all of the data to this string object

        'Assign the multi-value tokens to List(Of Objects)...
        Dim EmployeeNames As List(Of Object) = Me.GetTokenValue("EmployeeName")
		Dim GLAccounts As List(Of Object) = Me.GetTokenValue("GLAccount")
		Dim JobNames As List(Of Object) = Me.GetTokenValue("JobName")
		Dim JobNumbers As List(Of Object) = Me.GetTokenValue("JobNo")
		Dim RegularHrsWorked As List(Of Object) = Me.GetTokenValue("RegHrsWorked")

        'Since all multi-value lists contain the same number of lines we can get that
        'count from the EmployeeNames list...
        'Note: The Lists are zero based so the actual count of items is 1 less than 
        'the actual list count...
        Dim numRows As Integer = EmployeeNames.Count - 1

        'Here is the header row...
        headerRow = "Employee Name,GL Account,Rate,Job Name,Job#,Reg Hrs Worked,Over Time Hrs Worked,Deduction Amount,Deduction Code(s)"
        csvData.AppendLine(headerRow)

        'Here are the data rows...
        'Note: The first item is index 0...
        For Index As Integer = 0 To numRows
            csvData.AppendLine(EmployeeNames(Index) & "," & GLAccounts(Index) & "," & JobNames(Index) & "," & JobNumbers(Index) & "," & RegularHrsWorked(Index))
        Next Index

        'Now write the CSV file and close the streamwriter...
        csvWriter.Write(csvData)
        csvWriter.Close

        'Cleanup...
        csvWriter = Nothing
        csvData = Nothing

    Catch ex As Exception
        'Send any error messages to workflow so they can be tracked...
        WorkflowApi.TrackError(ex.message)

    End Try

One final note; the number of columns you are writing in the data rows is _not_ the number of rows that are in the header row so your final CSV format will be incorrect.  Pad the datarow AppendLine with the appropriate number of commas to get the column count expected.

Let us know if this resolved the issues!

 

5 0

Replies

replied on May 16, 2015 Show version history

Edgar - Without having the data it is a little difficult to diagnose but I would try several things related to the code and debugging. 

 

If I had to guess where the problem is it is in line 39.  Are you sure each multi-value field that you are referencing has the correct and same number of items in them?  i.e. if the 'EmployeeName' multi-value field has 12 items then the 'GL Account' multi-value field will need to have 12 items as well.  If not the script will throw and error when you try to reference an item index greater than the total number of items in the list.

 

A couple of thoughts related to the code that will help in debugging;

  • When using VB the preferential string concatenator symbol is the & not the + (as in C).  Although the VB compiler will accept the + it can introduce some ambiguity when trying to concatenate numeric string values (as you are in line 39).
  • I have not tried to do a TokenReplace on an entire concatenated string as you are in line 39.  (Although I would imagine that it does work)  Instead I would do a TokenReplace on each one of the column values, storing each column value to a unique variable, and then concatenate the column variables into the dataRow variable for writing to the CSV file.  (At least during the debugging stage.  If you want to try to improve performance after debugging the script then perhaps your method would be faster?)

 

Related to debugging;

  • If you can determine which which document is causing the error then you can run your code in the script editor against that specific document as the Starting Entry.  You can also open the Watch Window in the script editor and see the values being assigned to the variables as the code runs.
  • I would look for which rowNum (Index) in the For Next loop might be throwing the error, and also which multi-value field might have a fewer items than expected. 

 

 

3 0
replied on May 18, 2015 Show version history

Hi Cliff,

Thank you for the suggestions! I've done some updates to my code, but the issue persists. Please note that now I've created separate arrays for each token, and in this particular example I'm not printing them to the csv, as I simply wanted to test it out:

Imports System
Imports System.Collections.Generic
Imports System.ComponentModel
Imports System.Data
Imports System.Data.SqlClient
Imports System.Text

Namespace WorkflowActivity.Scripting.Script
   '''<summary>
    '''Provides one or more methods that can be run when the workflow scripting activity is performed.
    '''</summary>
    Public Class Script1
        Inherits ScriptClass90
        '''<summary>
        '''This method is run when the activity is performed.
        '''</summary>
        Protected Overrides Sub Execute()

    'Wrap the code in a Try/Catch to catch any errors...
    Try
       'Instantiate a new streamwriter...
        Dim fileName as String
        Dim numRows As Integer = GetTokenValue("RowCount")
        Dim regDate as Date = Date.Now()
        Dim strDate as String = regDate.ToString("ddMMMyyyy")
        fileName = TokenReplace("Weekly Report - From "+"%(DateTokenCalculator_1WeekAgo#""MM-dd-yyyy""#) To "+"%(Date#""MM-dd-yyyy""#)")
        My.Computer.FileSystem.CreateDirectory("D:\TestData\"+strDate+"\")
        Dim csvWriter As New System.IO.StreamWriter("D:\TestData\"+strDate+"\"+fileName+".csv")
        Dim headerRow As String 'Will hold the header row values
        Dim dataRow As String   'Will hold the data row values
        Dim csvData As New StringBuilder    'We will append all of the data to this string object
		Dim EmployeeNames() as String
		Dim GLAccounts() as String
		Dim JobNames() as String
		Dim JobNumbers() as String
		Dim RegularHrsWorked() as String


        'Here is the header row...
        headerRow = "Employee Name,GL Account,Rate,Job Name,Job#,Reg Hrs Worked,Over Time Hrs Worked,Deduction Amount,Deduction Code(s)"
        csvData.AppendLine(headerRow)

        'Here is the first data row...
        For Index As Integer = 0 To numRows
			EmployeeNames(Index) = Cstr(TokenReplace("%(EmployeeName_All#["+Index+"]#)"))
			GLAccounts(Index) = Cstr(TokenReplace("%(GLAccount_All#["+Index+"]#)"))
			JobNames(Index) = Cstr(TokenReplace("%(JobName_All#["+Index+"]#)"))
			JobNumbers(Index) = Cstr(TokenReplace("%(JobNo_All#["+Index+"]#)"))
			RegularHrsWorked(Index) = Cstr(TokenReplace("%(RegHrsWorked_All#["+Index+"]#)"))

            Next Index
			
		'dataRow = TokenReplace("%(EmployeeName_All#["+Index.ToString+"]#)"+","+"%(GLAccount_All#["+Index.ToString+"]#)"+",,"+"%(JobName_All#["+Index.ToString+"]#)"+","+"%(JobNo_All#["+Index.ToString+"]#)"+","+"%(RegHrsWorked_All#["+Index.ToString+"]#)"+",,,")
        'csvData.AppendLine(dataRow)

        'dataRow = TokenReplace("%(RetrieveFieldValues_Template Name)"+","+"%(RetrieveFieldValues_Account - Billing Address)"+","+"%(RetrieveFieldValues_Account - Drivers License)"+","+"%(RetrieveFieldValues_Account - Name)"+","+"%(RetrieveFieldValues_Account - Payment Method)"+","+"%(RetrieveFieldValues_Account - Relation)"+","+"%(RetrieveFieldValues_Account - SSN)"+","+"%(RetrieveFieldValues_Emergency - Contact)"+","+"%(RetrieveFieldValues_Emergency - Doctor)"+","+"%(RetrieveFieldValues_Emergency - Doctor's Phone)"+","+"%(RetrieveFieldValues_Emergency - Phone)"+","+"%(RetrieveFieldValues_Emergency - Relation)"+","+"%(RetrieveFieldValues_Health History - additional surgeries)"+","+"%(RetrieveFieldValues_Health History - Alergies)"+","+"%(RetrieveFieldValues_Health History - diseases/medical conditions/procedures_All)"+","+"%(RetrieveFieldValues_Health History - Exercise?)"+","+"%(RetrieveFieldValues_Health History - how many weeks pregnant?)"+","+"%(RetrieveFieldValues_Health History - how much exercise?)"+","+"%(RetrieveFieldValues_Health History - nursing?)"+","+"%(RetrieveFieldValues_Health History - pregnant?)"+","+"%(RetrieveFieldValues_Health History - smoker how often?)"+","+"%(RetrieveFieldValues_Health History - Smoker?)"+","+"%(RetrieveFieldValues_Health History - Supplements or vitamins?)"+","+"%(RetrieveFieldValues_Health History - taking any medications?_All)"+","+"%(RetrieveFieldValues_Health History - taking birth control?)"+","+"%(RetrieveFieldValues_Insurance - Co. Name_All)"+","+"%(RetrieveFieldValues_Insurance - Group_All)"+","+"%(RetrieveFieldValues_Insurance - Insured Date of Birth_All)"+","+"%(RetrieveFieldValues_Insurance - Insured Name_All)"+","+"%(RetrieveFieldValues_Insurance - Insured SSN_All)"+","+"%(RetrieveFieldValues_Insurance - Phone_All)"+","+"%(RetrieveFieldValues_Insurance - Relation_All)"+","+"%(RetrieveFieldValues_Patient - Age)"+","+"%(RetrieveFieldValues_Patient - Date of Birth)"+","+"%(RetrieveFieldValues_Patient - Email Address_All)"+","+"%(RetrieveFieldValues_Patient - Employment Status)"+","+"%(RetrieveFieldValues_Patient - Ethnicity)"+","+"%(RetrieveFieldValues_Patient - First Name)"+","+"%(RetrieveFieldValues_Patient - Gender)"+","+"%(RetrieveFieldValues_Patient - Last Name)"+","+"%(RetrieveFieldValues_Patient - Mailing Address)"+","+"%(RetrieveFieldValues_Patient - Marital Status)"+","+"%(RetrieveFieldValues_Patient - Middle Initial)"+","+"%(RetrieveFieldValues_Patient - Multi-Racial)"+","+"%(RetrieveFieldValues_Patient - Phone)"+","+"%(RetrieveFieldValues_Patient - Preferred Language)"+","+"%(RetrieveFieldValues_Patient - Race)"+","+"%(RetrieveFieldValues_Patient - SSN)"+","+"%(RetrieveFieldValues_Patient - Verification answer)"+","+"%(RetrieveFieldValues_Patient - Verification question)"+","+"%(RetrieveFieldValues_Visit - Are you in pain?)"+","+"%(RetrieveFieldValues_Visit - chiropractor clinic or doctor name)"+","+"%(RetrieveFieldValues_Visit - chiropractor clinic phone)"+","+"%(RetrieveFieldValues_Visit - Condition getting worse?)"+","+"%(RetrieveFieldValues_Visit - Condition interfering how)"+","+"%(RetrieveFieldValues_Visit - Condition interfering with)"+","+"%(RetrieveFieldValues_Visit - ever been threated where?)"+","+"%(RetrieveFieldValues_Visit - ever been treated by a physician for this condition?)"+","+"%(RetrieveFieldValues_Visit - ever been treated by chiropractor)"+","+"%(RetrieveFieldValues_Visit - Injured Area(s)_All)"+","+"%(RetrieveFieldValues_Visit - Injured during)"+","+"%(RetrieveFieldValues_Visit - Injured when?)"+","+"%(RetrieveFieldValues_Visit - Injured where?)"+","+"%(RetrieveFieldValues_Visit - Pain Level)"+","+"%(RetrieveFieldValues_Visit - Reason for Visit)"+","+"%(RetrieveFieldValues_Visit - something similar explain)"+","+"%(RetrieveFieldValues_Visit - something similar happened before?)"+","+"%(RetrieveFieldValues_Visit - what happened?)")
        'csvData.AppendLine(dataRow)

        'Here is the second data row...
        'dataRow = "Value 1, Value 2, Value 3, Value 4"
        'csvData.AppendLine(dataRow)

        'Now write the CSV file and close the streamwriter...
        csvWriter.Write(csvData)
        csvWriter.Close

        'Cleanup...
        csvWriter = Nothing
        csvData = Nothing

    Catch ex As Exception
        'Send any error messages to workflow so they can be tracked...
        WorkflowApi.TrackError(ex.message)

    End Try

End Sub

    End Class
End Namespace

To answer some of your questions:

- Yes, I'm sure each multi value token has the same number of items. I've attached on the original post a zip file containing the actual workflow with my test files (change extension from txt to zip). 

- I have already tried this code with some static values before, and the replaceToken function works well in a sentence like the one in my original code, however I've followed your suggestion and used it on separate variables.

Again, any help is greatly appreciated.

0 0
replied on May 18, 2015

Have you tried testing your script in the Script Editor? It might give you the line number where it fails or at least you can track your values for individual iterations.

0 0
replied on May 18, 2015

Hi Miruna,

Yes I did, and i added the same values i have in the metadata but it did not fail. I tested it with the first code I posted. 

0 0
SELECTED ANSWER
replied on May 18, 2015 Show version history

Edgar - Once I imported the workflow and documents you provided I can see what you are trying to do.  There were several errors but once corrected I can see a created CSV with document data.

The primary issues were that the RowCount token that you were passing to the script did not actually contain a numeric value but a string value of an EmployeeName. That is what was throwing the error you first reported.

After looking at the full workflow I realized that you do not need to do any TokenReplace on the passed multi-value string fields as the tokens have already been resolved by the 'RetrieveFieldValues' activity!

So here is the modified script;

    Try
        Dim strDate as String = Date.Now().ToString("ddMMMyyyy")
        Dim folderName As String = "D:\TestData\" & strDate & "\"
        Dim fileName As String = TokenReplace("Weekly Report - From "+"%(DateTokenCalculator_1WeekAgo#""MM-dd-yyyy""#) To "+"%(Date#""MM-dd-yyyy""#)")
        
        'Create the folder...
        My.Computer.FileSystem.CreateDirectory(folderName)
        
        'Open the streamwriter...
        Dim csvWriter As New System.IO.StreamWriter(folderName & fileName & ".csv")
        Dim headerRow As String 'Will hold the header row values
        Dim csvData As New StringBuilder    'We will append all of the data to this string object

        'Assign the multi-value tokens to List(Of Objects)...
        Dim EmployeeNames As List(Of Object) = Me.GetTokenValue("EmployeeName")
		Dim GLAccounts As List(Of Object) = Me.GetTokenValue("GLAccount")
		Dim JobNames As List(Of Object) = Me.GetTokenValue("JobName")
		Dim JobNumbers As List(Of Object) = Me.GetTokenValue("JobNo")
		Dim RegularHrsWorked As List(Of Object) = Me.GetTokenValue("RegHrsWorked")

        'Since all multi-value lists contain the same number of lines we can get that
        'count from the EmployeeNames list...
        'Note: The Lists are zero based so the actual count of items is 1 less than 
        'the actual list count...
        Dim numRows As Integer = EmployeeNames.Count - 1

        'Here is the header row...
        headerRow = "Employee Name,GL Account,Rate,Job Name,Job#,Reg Hrs Worked,Over Time Hrs Worked,Deduction Amount,Deduction Code(s)"
        csvData.AppendLine(headerRow)

        'Here are the data rows...
        'Note: The first item is index 0...
        For Index As Integer = 0 To numRows
            csvData.AppendLine(EmployeeNames(Index) & "," & GLAccounts(Index) & "," & JobNames(Index) & "," & JobNumbers(Index) & "," & RegularHrsWorked(Index))
        Next Index

        'Now write the CSV file and close the streamwriter...
        csvWriter.Write(csvData)
        csvWriter.Close

        'Cleanup...
        csvWriter = Nothing
        csvData = Nothing

    Catch ex As Exception
        'Send any error messages to workflow so they can be tracked...
        WorkflowApi.TrackError(ex.message)

    End Try

One final note; the number of columns you are writing in the data rows is _not_ the number of rows that are in the header row so your final CSV format will be incorrect.  Pad the datarow AppendLine with the appropriate number of commas to get the column count expected.

Let us know if this resolved the issues!

 

5 0
replied on May 18, 2015

Edgar - Had a random thought after I posted the code above.  I noticed that the EmployeeName values that you are writing to the CSV contain a comma character (LastName, FirstName).  Commas are reserved characters in a CSV file and cannot be used in any column data as the CSV assumes that the Employee LastName is column 1 and the Employee FirstName is column 2.

You could try wrapping the column data in quotes so that the column contents would be something like "Primmer, Cliff", "4 Rivers HQ Office Addition", "150111" ,"40"

You could also strip the comma out of the EmployeeName field before you write it to the CSV.

 

 

 

1 0
replied on February 3, 2016

Hi,

 

I have been directed to this post and it is great except i am having one issue that i cant seem to resolve

When i set the tokens such as :-

Dim DocumentName As List(Of Object) =Me.GetTokenValue("DocName")

and then run any scripts i am always getting and error

Do i have to format the tokens in a certain way to allow them to be built as a List?

The tokens are from retrieval of text fields?

 

Hope someone can help?

Cliff can you help?

 

Thanks in advance

Regards

0 0
replied on February 3, 2016

Mark, I think the problem here is that the token "DocName" is not a multi-value token and thus cannot be cast as a list. In your Workflow make sure you have the "Allow token to have multiple values" checked.

1 0
replied on February 3, 2016

Hi Chris,

 

Thanks so much for this. I already had this ticked but it pointed me to the fact that i was overwriting my tokens arather than ammending them.

Regards

 

Mark

0 0
replied on December 10, 2019

Another 'non-code' option is to load the multivalue token into a single value token. Then you can use the 'indexing' options to decide how you want to break up each of the lines of your CSV. 

This makes it so your workflows can be maintained by people who don't know VB or C#, but are comfortable with Workflow Designer.



 

1 0
replied on February 3, 2021

I added a comma in the Value between the brackets to get a comma between the values.

%(Multi_Value_Token#[,]#)

0 0
replied on May 9, 2016

Hi Guys,

 

This is great! This was my very first script attempt and I managed to get it working (for a non programmer), so a big thanks for that. I had tried a few others I have seen on LF Answers but had no luck with them. 

 

My intentions are to add more 2 more bits of functionality and not really sure if I have to create 2 separate script activities or add to the script above?

1. Save the CSV file into Laserfiche (will likely need a Path Token and File name token. Metadata not essential at this stage.

2. Email the CSV file. This may not be required if the 1st script can save to Laserfiche, I could use the "Search Repository" and "Email" Workflow activities.

 

thanks

Grant

1 0
replied on May 18, 2015

Hi Cliff!

Thank you very much for your help! As you mentioned, the issue was on the workflow at the rowCount token. I could have sworn I had enabled the value count function. After making that change, even my original code worked!

Thank you again!

0 0
replied on May 18, 2015

Good!  Glad it is working!

0 0
replied on May 2, 2019

If my data has comma like in employee name, it is spitting in two columns. I want it to spit in 1 column in csv file as comma is part of data. How do I do that please? Thanks.

 

Priya

0 0
replied on May 2, 2019

Priya,

I would try to wrap the employee name in double quotes either when you are adding it to the employeeName multivalue token or when you are actually writing the value out to the CSV.

The app opening the CSV should then ignore the comma between the last and first names and instead interpret the column as a single value.

1 0
replied on May 3, 2019

Thanks

0 0
You are not allowed to follow up in this post.

Sign in to reply to this post.