Skip to main content

Sheets in PouchDB

PouchDB is a pure JavaScript database with built-in synchronization features and offline support.

SheetJS is a JavaScript library for reading and writing data from spreadsheets.

This demo uses PouchDB and SheetJS to export database snapshots to spreadsheets and import bulk data from workbooks. We'll explore the subtleties of processing arrays of objects to mesh with both libraries.

The "Complete Example" section imbues the official "Todos" demo with the ability to export the list to XLSX workbooks.

Tested Deployments

This demo was tested in the following environments:

PouchDBDate
8.0.12024-05-04
7.3.12024-05-04
6.4.32024-05-04
5.4.52024-05-04
4.0.32024-05-04
3.6.02024-05-04

Integration Details

SheetJS CE offers standalone scripts, NodeJS modules, ESM modules, and other scripts. The "Installation" section covers a number of common deployment scenarios.

PouchDB ships with standalone scripts for browser use and NodeJS modules for use in server-side scripts1.

The PouchDB constructor returns a Database object.

Importing Data

Database#bulkDocs2 is the standard approach for bulk data import. The method accepts "arrays of objects" that can be generated through the SheetJS sheet_to_json3 method.

If rows do not include the _id parameter, the database will automatically assign an ID per row. It is strongly recommended to generate the _id directly.

This method starts from a SheetJS workbook object4 and uses data from the first sheet. read and readFile5 can generate workbook objects from files.

async function push_first_sheet_to_pouchdb(db, wb, _id_) {
/* get first worksheet */
const ws = wb.Sheets[wb.SheetNames[0]];

/* generate array of objects */
const aoo = XLSX.utils.sheet_to_json(ws);

/* if a prefix is specified, add a unique _id to each row based on index */
if(typeof _id_ == "string") aoo.forEach((row, idx) => row._id = _id_ + idx);

/* perform query */
return await db.bulkDocs(aoo);
}

Existing data can be erased with Database#destroy.

Exporting Data

Database#allDocs6 is the standard approach for bulk data export. Generated row objects have additional _id and _rev keys that should be removed.

After removing the PouchDB internal fields, the SheetJS json_to_sheet7 method can generate a worksheet. Other utility functions8 can construct a workbook. The workbook can be exported with the SheetJS writeFile9 method:

function export_pouchdb_to_xlsx(db) {
/* fetch all rows, including the underlying data */
db.allDocs({include_docs: true}, function(err, doc) {

/* pull the individual data rows */
const aoo = doc.rows.map(r => {
/* `rest` will include every field from `r` except for _id and _rev */
const { _id, _rev, ...rest } = r;
return rest;
});

/* generate worksheet */
const ws = XLSX.utils.json_to_sheet(aoo);

/* generate workbook and export */
const wb = XLSX.utils.book_new();
XLSX.utils.book_append_sheet(wb, ws, "Sheet1");
XLSX.writeFile(wb, "SheetJSPouch.xlsx");
});
}

json_to_sheet expects an array of "flattened" objects where each value is a simple data type that can be stored in a spreadsheet cell. If document objects have a nested structure, integration code should post-process the data.

"Export Tutorial" processes data from an API and computes a few text values from the nested data.

Complete Example

  1. Download the "Working Version" from the Getting Started guide.

The ZIP file should have MD5 checksum ac4da7cb0cade1be293ba222462f109c:

curl -LO https://github.com/nickcolley/getting-started-todo/archive/master.zip
md5sum master.zip || md5 master.zip

If the download is unavailable, a mirror is available at https://docs.sheetjs.com/pouchdb/master.zip :

curl -LO https://docs.sheetjs.com/pouchdb/master.zip
md5sum master.zip || md5 master.zip

The second command will display the checksum:

ac4da7cb0cade1be293ba222462f109c  master.zip
  1. Unzip the master.zip file and enter the folder:
unzip master.zip
cd getting-started-todo-master
  1. Edit index.html to reference the SheetJS library and add a button:
index.html (add highlighted lines)
  <body>
<script src="https://cdn.sheetjs.com/xlsx-0.20.3/package/dist/xlsx.full.min.js"></script>
<button id="xport">Export!</button>
<section id="todoapp">
  1. Near the end of index.html, look for a script tag referencing a CDN:
index.html (find line)
    <script src="//cdn.jsdelivr.net/pouchdb/3.2.0/pouchdb.min.js"></script>

Upgrade PouchDB by changing the src attribute to the production build10:

index.html (replace line)
    <script src="//cdn.jsdelivr.net/npm/[email protected]/dist/pouchdb.min.js"></script>
  1. Just before the end of js/app.js, add a click event listener:
js/app.js (add highlighted lines)
  if (remoteCouch) {
sync();
}

document.getElementById("xport").addEventListener("click", function() {
db.allDocs({include_docs: true, descending: true}, function(err, doc) {
const aoo = doc.rows.map(r => {
const { _id, _rev, ... rest } = r.doc;
return rest;
});
const ws = XLSX.utils.json_to_sheet(aoo);
const wb = XLSX.utils.book_new(); XLSX.utils.book_append_sheet(wb, ws, "Sheet1");
XLSX.writeFile(wb, "SheetJSPouch.xlsx");
});
});
})();

The demo UI reads the todo items in descending order:

js/app.js
    //------------------------------VVVVVVVVVVVVVVVV (descending order)
db.allDocs({include_docs: true, descending: true}, function(err, doc) {
redrawTodosUI(doc.rows);
});

The new callback function also specifies descending: true to ensure that the order of todo items in the export matches the list displayed in the webpage.

  1. Start a local web server:
npx http-server .

The command will display a URL (typically http://localhost:8080) which can be opened in a web browser.

Testing

  1. Access the URL from step 5 with a web browser.

  2. Add two items "js" and "Sheet". Mark "Sheet" as completed. The page should look like the following screenshot:

todos screenshot

  1. Click the "Export!" text at the top of the page. The site should create an export named "SheetJSPouch.xlsx"

  2. Open the file in a spreadsheet editor. It should match the following table:

titlecompleted
SheetTRUE
jsFALSE

Footnotes

  1. See "Setting up PouchDB" in the PouchDB documentation.

  2. See "Create/update a batch of documents" in the PouchDB API documentation

  3. See sheet_to_json in "Utilities"

  4. See "SheetJS Data Model"

  5. See read in "Reading Files"

  6. See "Fetch a batch of documents" in the PouchDB API documentation

  7. See json_to_sheet in "Utilities"

  8. See "Workbook Helpers" in "Utilities" for details on book_new and book_append_sheet.

  9. See writeFile in "Writing Files"

  10. The "Quick Start" section of "Download" in the PouchDB website describes the recommended CDN for PouchDB scripts.