This page is for the specific failure mode where an ExcelJS workflow can create or update formula cells, but the Node.js process also needs the calculated result after changing inputs.
That is a runtime problem, not just a file-writing problem.
Use ExcelJS when the product is an .xlsx file that Excel, LibreOffice, or
another spreadsheet application will open and calculate later.
Use a formula runtime when the service needs to write inputs and read the
recalculated values before returning a response. @bilig/headless is one
option for the narrower case where the workbook can live as a TypeScript
WorkPaper document with JSON persistence and verified readback.
Spreadsheet files can contain formula text and cached results. A file library can preserve those records, and some libraries let you supply the cached result yourself. That does not mean the library recalculated the dependency graph after a service changed an input cell.
A common ExcelJS trap looks like this:
workbook.calcProperties.fullCalcOnLoad = true
worksheet.getCell('A1').value = 15
console.log(worksheet.getCell('C1').value)
// { formula: 'A1+B1', result: 20 }
If C1 originally cached 20, setting fullCalcOnLoad does not make
worksheet.getCell('C1').value become 25 inside the same Node.js process.
That flag tells a spreadsheet application to recalculate when it opens the
file. It is a file instruction, not an in-process calculation engine.
The difference matters in production:
If step 3 happens later in Excel, the backend never owned the decision.
| Job | Better starting point |
|---|---|
| Generate an XLSX report with styles, sheets, tables, and formula strings | ExcelJS |
| Open a file later in Excel and let Excel calculate formulas | ExcelJS |
| Preserve formula records and cached values from an existing workbook | ExcelJS or SheetJS-style tooling |
| Recalculate workbook formulas inside a Node.js request, job, or agent tool | A formula runtime such as @bilig/headless |
| Persist formula-backed state as JSON and verify it after restore | @bilig/headless WorkPaper |
Install the runtime in a scratch project:
mkdir exceljs-recalc-eval
cd exceljs-recalc-eval
npm init -y
npm pkg set type=module
npm install @bilig/headless
npm install -D tsx typescript @types/node
Create recalculate.ts:
import {
WorkPaper,
createWorkPaperFromDocument,
exportWorkPaperDocument,
parseWorkPaperDocument,
serializeWorkPaperDocument,
} from '@bilig/headless'
type NumericCell = {
value: number
}
function readNumber(cell: unknown, label: string): number {
if (typeof cell === 'object' && cell !== null && typeof (cell as NumericCell).value === 'number') {
return (cell as NumericCell).value
}
throw new Error(`Expected ${label} to be numeric, got ${JSON.stringify(cell)}`)
}
const workbook = WorkPaper.buildFromSheets({
Inputs: [
['Metric', 'Value'],
['Units', 100],
['Unit price', 49],
['Discount', 0.1],
],
Quote: [
['Metric', 'Value'],
['Net total', '=Inputs!B2*Inputs!B3*(1-Inputs!B4)'],
],
})
const inputs = workbook.getSheetId('Inputs')
const quote = workbook.getSheetId('Quote')
if (inputs === undefined || quote === undefined) {
throw new Error('Expected Inputs and Quote sheets')
}
const netTotalCell = { sheet: quote, row: 1, col: 1 }
const before = readNumber(workbook.getCellValue(netTotalCell), 'before')
workbook.setCellContents({ sheet: inputs, row: 3, col: 1 }, 0.25)
const after = readNumber(workbook.getCellValue(netTotalCell), 'after')
const serialized = serializeWorkPaperDocument(exportWorkPaperDocument(workbook, { includeConfig: true }))
const restored = createWorkPaperFromDocument(parseWorkPaperDocument(serialized))
const restoredQuote = restored.getSheetId('Quote')
if (restoredQuote === undefined) {
throw new Error('Expected restored Quote sheet')
}
const afterRestore = readNumber(restored.getCellValue({ sheet: restoredQuote, row: 1, col: 1 }), 'after restore')
console.log({
before,
after,
afterRestore,
verified: before === 4410 && after === 3675 && afterRestore === after,
})
Run it:
npx tsx recalculate.ts
Expected output:
{ "before": 4410, "after": 3675, "afterRestore": 3675, "verified": true }
That verifies the part ExcelJS is not designed to own: an input changed, dependent formulas recalculated in the same Node.js process, and the persisted document restored with the same computed result.
The honest architecture is to keep file generation and formula runtime separate:
.xlsx files, styling, worksheets, and reports.@bilig/headless for the formula-backed business state your service
must trust immediately.Do not mix these responsibilities silently. A cached value in a file is not the same as recalculated business state.
Do not choose @bilig/headless only to generate styled XLSX files.
Do not choose it if a human can open the workbook in Excel before any business decision depends on the calculated value.
Do not choose it when you need full Excel compatibility across every formula, chart, pivot table, macro, or workbook artifact. Check the compatibility limits first.
If this saves you an ExcelJS recalculation workaround, star the repository so the project is easier for the next backend developer to find: https://github.com/proompteng/bilig/stargazers.