@ -26,7 +26,7 @@ can be installed with the following command:
@@ -26,7 +26,7 @@ can be installed with the following command:
npm install papaparse
If you don't want to use npm, [papaparse.min.js](https://unpkg.com/papaparse@latest/papaparse.min.js) can be downloaded to your project source.
If you don't want to use npm, [papaparse.min.js](https://github.com/mholt/PapaParse/blob/master/papaparse.min.js) can be downloaded to your project source.
Homepage & Demo
@ -39,7 +39,9 @@ To learn how to use Papa Parse:
@@ -39,7 +39,9 @@ To learn how to use Papa Parse:
- [Documentation](http://papaparse.com/docs)
The website is hosted on [Github Pages](https://pages.github.com/). Its content is also included in the docs folder of this repository. If you want to contribute on it just clone the master of this repository and open a pull request.
The website is hosted on on [Github Pages](https://pages.github.com/). If
you want to contribute just clone the gh-branch of this repository and
to alter the flow of parsing. Actions can be <code>"abort"</code> to skip this and all other files in the queue, <code>"skip"</code> to skip just this file, or <code>"continue"</code> to carry on (equivalent to returning nothing). <code>reason</code> can be a reason for aborting. <code>config</code> can be a modified <ahref="#config">configuration</a> for parsing just this file.
to alter the flow of parsing. Actions can be <code>"abort"</code> to skip this and all other files in the queue, <code>"skip"</code> to skip just this file, or <code>"continue"</code> to carry on (equivalent to returning nothing). <code>reason</code> can be a reason for aborting. <code>config</code> can be a modified <ahref="#config">configuration</a> for parsing just this file.</li>
</li>
<li>The <code>complete</code> callback shown here is executed after <i>all</i> files are finished and does not receive any data. Use the complete callback in <ahref="#config">config</a> for per-file results.</li>
skipEmptyLines: false, //other option is 'greedy', meaning skip delimiters, quotes, and whitespace.
columns: null //or array of strings
}
</code></pre>
</div>
<divclass="clear"></div>
<divclass="grid-100">
<h5>Unparse Config Options</h5>
</div>
<divclass="grid-100"style="overflow-x: auto;">
<table>
<tr>
<thstyle="width: 20%;">Option</th>
<thstyle="width: 80%;">Explanation</th>
</tr>
<tr>
<td>
<code>quotes</code>
</td>
<td>
If <code>true</code>, forces all fields to be enclosed in quotes. If an array of <code>true/false</code> values, specifies which fields should be force-quoted (first boolean is for the first column, second boolean for the second column, ...). A function that returns a boolean values can be used to determine the quotes value of a cell. This function accepts the cell value and column index as parameters. <br/>
Note that this option is ignored for <code>undefined</code>, <code>null</code> and <code>date-object</code> values. The option <code>escapeFormulae</code> also takes precedence over this.
</td>
</tr>
<tr>
<td><code>quoteChar</code></td>
<td>
The character used to quote fields.
</td>
</tr>
<tr>
<td><code>escapeChar</code></td>
<td>
The character used to escape <code>quoteChar</code> inside field values.
</td>
</tr>
<tr>
<td>
<code>delimiter</code>
</td>
<td>
The delimiting character. Multi-character delimiters are supported. It must not be found in <ahref="#readonly">Papa.BAD_DELIMITERS</a>.
</td>
</tr>
<tr>
<td>
<code>header</code>
</td>
<td>
If <code>false</code>, will omit the header row. If <code>data</code> is an array of arrays this option is ignored. If <code>data</code> is an array of objects the keys of the first object are the header row. If <code>data</code> is an object with the keys <code>fields</code> and <code>data</code> the <code>fields</code> are the header row.
</td>
</tr>
<tr>
<td>
<code>newline</code>
</td>
<td>
The character used to determine newline sequence. It defaults to <code>"\r\n"</code>.
</td>
</tr>
<tr>
<td>
<code>skipEmptyLines</code>
</td>
<td>
If <code>true</code>, lines that are completely empty (those which evaluate to an empty string) will be skipped. If set to <code>'greedy'</code>, lines that don't have any content (those which have only whitespace after parsing) will also be skipped.
</td>
</tr>
<tr>
<td>
<code>columns</code>
</td>
<td>
If <code>data</code> is an array of objects this option can be used to manually specify the keys (columns) you expect in the objects. If not set the keys of the first objects are used as column.
</td>
</tr>
<tr>
<td>
<code>escapeFormulae</code>
</td>
<td>
If <code>true</code>, field values that begin with <code>=</code>, <code>+</code>, <code>-</code>, <code>@</code>, <code>\t</code>, or <code>\r</code>, will be prepended with a <code>'</code> to defend against <ahref="https://owasp.org/www-community/attacks/CSV_Injection"target="_blank"rel="noopener">injection attacks</a>, because Excel and LibreOffice will automatically parse such cells as formulae. You can override those values by setting this option to a regular expression
</td>
</tr>
</table>
newline: "\r\n"
}</code></pre>
Set <code>quotes</code> to <code>true</code> to always enclose each field in quotes, or an array of true/false values correlating to specific to columns to force-quote. The character used to quote can be customized using <code>quoteChar</code>. The character used to escape the <code>quoteChar</code> within a field can be customized using <code>escapeChar</code>. The <code>delimiter</code> can be any valid delimiting character. The <code>newline</code> character(s) may also be customized. Setting <code>header</code> to <code>false</code> will omit the header row.
</li>
</ul>
</div>
<divclass="clear"></div>
@ -380,8 +289,8 @@ var csv = Papa.unparse([
@@ -380,8 +289,8 @@ var csv = Papa.unparse([
<divclass="grid-33">
<pre><codeclass="language-javascript">// Specifying fields and data explicitly
var csv = Papa.unparse({
"fields": ["Column 1", "Column 2"],
"data": [
fields: ["Column 1", "Column 2"],
data: [
["foo", "bar"],
["abc", "def"]
]
@ -431,7 +340,7 @@ var csv = Papa.unparse({
@@ -431,7 +340,7 @@ var csv = Papa.unparse({
quoteChar: '"',
escapeChar: '"',
header: false,
transformHeader: undefined,
trimHeaders: false,
dynamicTyping: false,
preview: 0,
encoding: "",
@ -441,16 +350,12 @@ var csv = Papa.unparse({
@@ -441,16 +350,12 @@ var csv = Papa.unparse({
@ -470,7 +375,7 @@ var csv = Papa.unparse({
@@ -470,7 +375,7 @@ var csv = Papa.unparse({
<code>delimiter</code>
</td>
<td>
The delimiting character. Leave blank to auto-detect from a list of most common delimiters, or any values passed in through <code>delimitersToGuess</code>. It can be a string or a function. If a string, it can be of any length (so multi-character delimiters are supported). If a function, it must accept the input as first parameter and it must return a string which will be used as delimiter. In both cases it cannot be found in <ahref="#readonly">Papa.BAD_DELIMITERS</a>.
The delimiting character. Leave blank to auto-detect from a list of most common delimiters. It can be a string or a function. If string, it must be one of length 1. If a function, it must accept the input as first parameter and it must return a string which will be used as delimiter. In both cases it cannot be found in <ahref="#readonly">Papa.BAD_DELIMITERS</a>.
</td>
</tr>
<tr>
@ -507,11 +412,10 @@ var csv = Papa.unparse({
@@ -507,11 +412,10 @@ var csv = Papa.unparse({
</tr>
<tr>
<td>
<code>transformHeader</code>
<code>trimHeaders</code>
</td>
<td>
A function to apply on each header. Requires <code>header</code> to be <code>true</code>. The function receives the header as its first argument and the index as second.<br>
Only available starting with version 5.0.
If true leading/trailing spaces will be trimed from headers.
</td>
</tr>
<tr>
@ -519,7 +423,7 @@ var csv = Papa.unparse({
@@ -519,7 +423,7 @@ var csv = Papa.unparse({
<code>dynamicTyping</code>
</td>
<td>
If true, numeric and boolean data will be converted to their type instead of remaining strings. Numeric data must conform to the definition of a decimal literal. Numerical values greater than <code>2^53</code> or less than <code>-2^53</code> will not be converted to numbers to preserve precision. European-formatted numbers must have commas and dots swapped. If also accepts an object or a function. If object it's values should be a boolean to indicate if dynamic typing should be applied for each column number (or header name if using headers). If it's a function, it should return a boolean value for each field number (or name if using headers) which will be passed as first argument.
If true, numeric and boolean data will be converted to their type instead of remaining strings. Numeric data must conform to the definition of a decimal literal. European-formatted numbers must have commas and dots swapped. If also accepts an object or a function. If object it's values should be a boolean to indicate if dynamic typing should be applied for each column number (or header name if using headers). If it's a function, it should return a boolean value for each field number (or name if using headers) which will be passed as first argument.
</td>
</tr>
<tr>
@ -527,7 +431,7 @@ var csv = Papa.unparse({
@@ -527,7 +431,7 @@ var csv = Papa.unparse({
<code>preview</code>
</td>
<td>
If > 0, only that many rows will be parsed.
If > 0, only that many rows will be parsed.
</td>
</tr>
<tr>
@ -543,7 +447,7 @@ var csv = Papa.unparse({
@@ -543,7 +447,7 @@ var csv = Papa.unparse({
<code>worker</code>
</td>
<td>
Whether or not to use a <ahref="/faq#workers">worker thread</a>. Using a worker will keep your page reactive, but may be slightly slower.
Whether or not to use a <ahref="/faq#workers">worker thread</a>. Using a worker will keep your page reactive, but may be slightly slower. Web Workers also load the entire Javascript file, so be careful when <ahref="/faq#combine">combining other libraries</a> in the same file as Papa Parse. Note that worker option is only available when parsing files and not when converting from JSON to CSV.
</td>
</tr>
<tr>
@ -596,27 +500,6 @@ var csv = Papa.unparse({
@@ -596,27 +500,6 @@ var csv = Papa.unparse({
If true, this indicates that the string you passed as the first argument to <code>parse()</code> is actually a URL from which to download a file and parse its contents.
</td>
</tr>
<tr>
<td>
<code>downloadRequestHeaders</code>
</td>
<td>
If defined, should be an object that describes the headers, example:
Use POST request on the URL of the download option. The value passed will be set as the body of the request.
</td>
</tr>
<tr>
<td>
<code>skipEmptyLines</code>
@ -633,14 +516,6 @@ var csv = Papa.unparse({
@@ -633,14 +516,6 @@ var csv = Papa.unparse({
A callback function, identical to step, which activates streaming. However, this function is executed after every <i>chunk</i> of the file is loaded and parsed rather than every row. Works only with local and remote files. Do not use both chunk and step callbacks together. For the function signature, see the documentation for the step function.
</td>
</tr>
<tr>
<td>
<code>chunkSize</code>
</td>
<td>
Overrides <code>Papa.LocalChunkSize</code> and <code>Papa.RemoteChunkSize</code>. See <ahref="#configurable">configurable</a> section to know the usage of both parameters.
</td>
</tr>
<tr>
<td>
<code>fastMode</code>
@ -670,15 +545,7 @@ var csv = Papa.unparse({
@@ -670,15 +545,7 @@ var csv = Papa.unparse({
<code>transform</code>
</td>
<td>
A function to apply on each value. The function receives the value as its first argument and the column number or header name when enabled as its second argument. The return value of the function will replace the value it received. The transform function is applied before dynamicTyping.
</td>
</tr>
<tr>
<td>
<code>delimitersToGuess</code>
</td>
<td>
An array of delimiters to guess from if the <code>delimiter</code> option is not set.
A function to apply on each value. The function receives the value as its first argument and the column number as its second argument. The return value of the function will replace the value it received. The transform function is applied before dynamicTyping.
</td>
</tr>
</table>
@ -861,7 +728,7 @@ var csv = Papa.unparse({
@@ -861,7 +728,7 @@ var csv = Papa.unparse({
<tr>
<td><code>Papa.BAD_DELIMITERS</code></td>
<td>
An array of characters that are not allowed as delimiters (<code>\r, \n, ", \ufeff</code>).
An array of characters that are not allowed as delimiters.
</td>
</tr>
<tr>
@ -882,6 +749,12 @@ var csv = Papa.unparse({
@@ -882,6 +749,12 @@ var csv = Papa.unparse({
Whether or not the browser supports HTML5 Web Workers. If false, <code>worker: true</code> will have no effect.
</td>
</tr>
<tr>
<td><code>Papa.SCRIPT_PATH</code></td>
<td>
The relative path to Papa Parse. This is automatically detected when Papa Parse is loaded synchronously. However, if you load Papa Parse asynchronously (e.g. with RequireJS), you need to set this variable manually in order to use Web Workers. (In those cases, this variable is <i>not</i> read-only and you should set it!)
</td>
</tr>
</table>
</div>
@ -939,7 +812,7 @@ var csv = Papa.unparse({
@@ -939,7 +812,7 @@ var csv = Papa.unparse({
<br><br>
Papa Parse by <ahref="https://twitter.com/mholt6">Matt Holt</a>
<h6id="combine">Can I put other libraries in the same file as Papa Parse?</h6>
<p>
Yes.
Yes, but then don't use the Web Worker feature unless your other dependencies are battle-hardened for worker threads. A worker thread loads an entire file, not just a function, so all those dependencies would be executed in an environment without a DOM and other <code>window</code> features. If any of those dependencies crash (<code>Cannot read property "defaultView" of undefined</code><ahref="https://github.com/mholt/PapaParse/issues/114">is</a><ahref="https://github.com/mholt/PapaParse/issues/163">common</a>), the whole worker thread will crash and parsing will not succeed.
</p>
@ -93,7 +96,7 @@
@@ -93,7 +96,7 @@
<h6id="async">Can Papa Parse be loaded asynchronously (after the page loads)?</h6>
<p>
Yes.
Yes. But if you want to use Web Workers, you'll need to specify the relative path to Papa Parse. To do this, set <ahref="/docs#readonly">Papa.SCRIPT_PATH</a> to the relative path of the Papa Parse file. In synchronous loading, this is automatically detected.
</p>
@ -206,7 +209,7 @@
@@ -206,7 +209,7 @@
<h6>Can I use a worker if I combine/concatenate my Javascript files?</h6>
<p>
Yes.
Probably not. It's safest to concatenate the rest of your dependencies and include Papa Parse in a seperate file. Any library that expects to have access to the <code>window</code> or DOM will crash when executed in a worker thread. Only put <ahref="/faq#combine">other libraries in the same file</a> if they are ready to be used in worker threads.
</p>
<h6>When should I use a worker?</h6>
@ -238,11 +241,6 @@
@@ -238,11 +241,6 @@
<p>
No. This would drastically slow down parsing, as it would require the worker to wait after every chunk for a "continue" signal from the main thread. But you <i>can</i> abort workers by calling <code>.abort()</code> on the parser that gets passed to your callback function.
</p>
<h6>I set worker:true and now I'm getting an error: "window is not defined." How do I fix it?</h6>
<p>
This is a fairly common issue with configuration and it appears to be related to the use of React or other third party tools. Since this is a configuration issue, the exact steps needed to solve it may vary. See <ahref="https://github.com/mholt/PapaParse/issues/655">Issue #655</a> on GitHub for a solution that worked for one person, and for links to other related issues.
</p>
</div>
</div>
</main>
@ -259,7 +257,7 @@
@@ -259,7 +257,7 @@
<br><br>
Papa Parse by <ahref="https://twitter.com/mholt6">Matt Holt</a>
<b><ahref="https://github.com/mholt/PapaParse/blob/master/docs/resources/js/lovers.js"class="add-lover-link subheader"><iclass="fa fa-plus-square"></i> Add your link (it's free)</a></b>
<b><ahref="https://github.com/mholt/PapaParse/blob/gh-pages/resources/js/lovers.js"class="add-lover-link subheader"><iclass="fa fa-plus-square"></i> Add your link (it's free)</a></b>
</div>
</div>
</section>
@ -521,7 +506,6 @@ var csv = Papa.unparse(yourData);</code></pre>
@@ -521,7 +506,6 @@ var csv = Papa.unparse(yourData);</code></pre>
<iclass="fa fa-book"></i> Documentation
</a>
</div>
</div>
</section>
@ -540,7 +524,7 @@ var csv = Papa.unparse(yourData);</code></pre>
@@ -540,7 +524,7 @@ var csv = Papa.unparse(yourData);</code></pre>
<br><br>
Papa Parse by <ahref="https://twitter.com/mholt6">Matt Holt</a>
@ -56,57 +56,27 @@ var peopleLovePapa = [
@@ -56,57 +56,27 @@ var peopleLovePapa = [
description:"uses Papa Parse in VisualEditor to help article editors effortlessly build data tables from text files."
},
{
link:"https://github.com/Nanofus/novel.js",
name:"Novel.js",
description:"is a text adventure framework that uses Papa Parse to enable user-friendly translations.",
quote:"Papa saves countless hours of work and makes reading large CSV files so easy!"
},
{
link:"https://mailcheck.co",
name:"Mailcheck.co",
description:"Mailcheck is email validation service. All emails usually stored in CSV's. We use Papa Parse to process data from our customers in browser",
quote:"Papa Parser allowed our customers to preview and process csv's in browser, without uploading them to server. It saves lots of time and space :)"
description:"created a video showing how to use Papa Parse and FileDrop.js to create a drag-and-drop CSV-JSON converter.",
quote:"It's often easy to convert data to CSV. With Papa, it's easy to turn that CSV into JSON."
},
{
link:"https://flatfile.io",
name:"Flatfile.io",
description:"is an add-in data importer for web-apps, providing the full UX to upload a spreadsheet, field match, and repair issues found during import.",
quote:"Papa is a core part of our importer, so much so that we're committed to helping maintain it!"
description:"created a simple regression test for Papa Parse.",
quote:"Papa's API is so intuitive, it took me no time to get it to work."
},
{
link:"https://familiohq.com",
name:"Familio",
description:"is a brand-new messaging app made specifically for busy families. Automatically align all family members when sending text messages to parents in the kindergarten or school or when planning your kids birthday parties.",
quote:"With Papa it was a joy to implement our tool for importing messages and places from external systems."
link:"https://www.appstax.com",
name:"Appstax",
description:"uses Papa Parse to import and export CSV data in their visual databrowser.",
quote:"Papa is a great for parsing CSV. And what a great tone of voice - love it!"
},
{
link:"https://monei.net",
name:"MONEI",
description:"Digital payments made easy.",
quote:"With Papa life became much easier for us to manage huge csv payments files of our merchants."
},
{
link:"https://moonmail.io",
name:"MoonMail",
description:"OmniChannel Communication Platform powered by AWS PinPoint",
quote:"Papa makes contact imports a plain sailing."
},
{
link:"https://apps.shopify.com/wholesaler",
name:"Wholesaler for Shopify",
description:"Shopify App to offer Wholesaling within one unique Shopify store",
quote:"Super fast bulk Wholesale product price uploads. Love Papa!."
},
{
link:"https://www.unnitmetaliya.com/sop-sample/",
name:"Visa SOP Sample",
description:"Providing free guide to international students.",
quote:"Use Papa Parse for many of side projects. Super fast and works all the time. Love it!"
},
{
link:"https://retool.com/",
name:"Retool",
description:"A remarkably fast way to build internal tools.",
quote:"Papa makes it easy for our users to customize CSV parsing to match their business logic."
link:"https://github.com/Nanofus/novel.js",
name:"Novel.js",
description:"is a text adventure framework that uses Papa Parse to enable user-friendly translations.",
quote:"Papa saves countless hours of work and makes reading large CSV files so easy!"
"description":"Fast and powerful CSV parser for the browser that supports web workers and streaming large files. Converts CSV to JSON and JSON to CSV.",
@ -965,39 +912,6 @@ var PARSE_TESTS = [
@@ -965,39 +912,6 @@ var PARSE_TESTS = [
errors:[]
}
},
{
description:"Custom transform accepts column number also",
input:'A,B,C\r\nd,e,f',
config:{
transform:function(value,column){
if(column%2){
value=value.toLowerCase();
}
returnvalue;
}
},
expected:{
data:[["A","b","C"],["d","e","f"]],
errors:[]
}
},
{
description:"Custom transform accepts header name when using header",
input:'A,B,C\r\nd,e,f',
config:{
header:true,
transform:function(value,name){
if(name==='B'){
value=value.toUpperCase();
}
returnvalue;
}
},
expected:{
data:[{'A':"d",'B':"E",'C':"f"}],
errors:[]
}
},
{
description:"Dynamic typing converts ISO date strings to Dates",
input:'ISO date,long date\r\n2018-05-04T21:08:03.269Z,Fri May 04 2018 14:08:03 GMT-0700 (PDT)\r\n2018-05-08T15:20:22.642Z,Tue May 08 2018 08:20:22 GMT-0700 (PDT)',
@ -1007,15 +921,6 @@ var PARSE_TESTS = [
@@ -1007,15 +921,6 @@ var PARSE_TESTS = [
errors:[]
}
},
{
description:"Dynamic typing skips ISO date strings ocurring in other strings",
input:'ISO date,String with ISO date\r\n2018-05-04T21:08:03.269Z,The date is 2018-05-04T21:08:03.269Z\r\n2018-05-08T15:20:22.642Z,The date is 2018-05-08T15:20:22.642Z',
config:{dynamicTyping:true},
expected:{
data:[["ISO date","String with ISO date"],[newDate("2018-05-04T21:08:03.269Z"),"The date is 2018-05-04T21:08:03.269Z"],[newDate("2018-05-08T15:20:22.642Z"),"The date is 2018-05-08T15:20:22.642Z"]],
errors:[]
}
},
{
description:"Blank line at beginning",
input:'\r\na,b,c\r\nd,e,f',
@ -1264,36 +1169,6 @@ var PARSE_TESTS = [
@@ -1264,36 +1169,6 @@ var PARSE_TESTS = [
errors:[]
}
},
{
description:"Pipe delimiter is guessed correctly choose avgFildCount max one",
notes:"Guessing the delimiter should work choose the min delta one and the max one",
config:{},
input:'a,b,c\na,b,c|d|e|f',
expected:{
data:[['a','b','c'],['a','b','c|d|e|f']],
errors:[]
}
},
{
description:"Pipe delimiter is guessed correctly when first field are enclosed in quotes and contain delimiter characters",
notes:"Guessing the delimiter should work if the first field is enclosed in quotes, but others are not",
description:"Returns empty rows when empty rows are passed and skipEmptyLines is false with headers",
@ -1882,87 +1698,7 @@ var UNPARSE_TESTS = [
@@ -1882,87 +1698,7 @@ var UNPARSE_TESTS = [
input:[{a:null,b:' '},{},{a:'1',b:'2'}],
config:{skipEmptyLines:'greedy',header:true},
expected:'a,b\r\n1,2'
},
{
description:"Column option used to manually specify keys",
notes:"Should not throw any error when attempting to serialize key not present in object. Columns are different than keys of the first object. When an object is missing a key then the serialized value should be an empty string.",
input:[{a:1,b:'2'},{},{a:3,d:'d',c:4,}],
config:{columns:['a','b','c']},
expected:'a,b,c\r\n1,2,\r\n\r\n3,,4'
},
{
description:"Column option used to manually specify keys with input type object",
notes:"Should not throw any error when attempting to serialize key not present in object. Columns are different than keys of the first object. When an object is missing a key then the serialized value should be an empty string.",