Fusionable JS is a JavaScript/TypeScript library designed for developers who work with Markdown as a content storage format
Fusionable is a JavaScript/TypeScript library designed for developers who work with Markdown as a content storage format. It provides a powerful and flexible API to parse, filter, and manage collections of Markdown files based on their metadata (frontmatter). With Fusionable, you can easily treat Markdown files as structured content collections, enabling efficient filtering, sorting, and retrieval for various applications, from documentation systems to content-driven websites.
Fusionable is still under development. We have the version 0.0.*. Feel free to use it in your experiment but not for production. Based on your feedback, we are working hard to make it stable and production-ready.
An example of using Fusionable for handling Markdown content:
let contents = new FusionCollection()
.loadFromDir(`./src/content`) //load markdown files from a directory
.filter("highlight") // filtering
.orderBy("date") // sorting markdown by `date` field
.limit(3) // limiting the result
.getItemsArray();
Once you read this README, you can explore more how to use Fusionable with SvelteKit, Nextjs, and Nuxt:
- SvelteKit: How to Build a Content-Driven Static Site with Markdown, SvelteKit and Fusionable
- Next.js: Building a website using Markdown content with Next.js App Router and Fusionable
- Nuxt: Build a static website with Markdown content, using Nuxt and Fusionable (server API approach)
Fusionable is a library that allows you to organize, filter, and query collections of Markdown files. It leverages frontmatter metadata in each file, enabling you to:
- Parse and extract content from Markdown files.
- Retrieve files or collections based on metadata fields.
- Sort and filter content collections dynamically with a fluent API.
- Use Markdown files as a structured content source for building dynamic content applications.
With Fusionable, Markdown files have become more than simple documents; they have become manageable, queryable data sources.
Example use cases for Fusionable
- Content-driven Websites: Manage and retrieve blog posts, news articles, or other structured content stored in Markdown format.
- Static Site Generation: Integrate Fusionable in static site generators to dynamically filter and sort Markdown content.
- Documentation Systems: Organize documentation sections by metadata like tags, dates, or custom categories.
To install the Fusionable JS package:
bun add fusionable
Or, if you are using NPM and Node, you can install it via the npm
command:
npm i fusionable
Then, you can create your Markdown files, for example, in the ./src/content
directory:
---
title: "Post One"
description: This is the first post
date: "2020-01-01"
highlight: true
tags: ["typescript", "markdown"]
cover: https://picsum.photos/seed/post-1/400/200
---
## Hello
Content of the *first post*.
You can parse, load, filter, and sort the content via:
import FusionCollection from "fusionable/FusionCollection"
let contents = new FusionCollection()
.loadFromDir(`./src/content`)
.filter('highlight')
.orderBy("date")
.limit(3)
.getItemsArray();
Or, you can simply get one content via the slug field in the Frontmatter:
const item = new FusionCollection()
.loadFromDir('./tests/data/')
.getOneBySlug('post-1');
The FusionCollection
class provides a flexible, chainable API for loading, filtering, sorting, limiting, and retrieving data. Below is a breakdown of the main methods available.
Loads content from a specified directory and initializes the collection with items found in that directory.
Parameter:
directoryPath
(string
): The path to the directory from which to load content. It returns: TheFusionCollection
instance allows for method chaining.
Example:
// loadFromDir(directoryPath: string): FusionCollection
let collection = new FusionCollection().loadFromDir(`./src/content`);
Adds a filter condition to refine the collection based on a specific field, comparison operator, and value.
Parameters:
field
: (string
) — The field to apply the filter on.operator
: (OperatorEnum
, optional) — The comparison operator. Defaults to strict equality (===
). Options include:- OperatorEnum.EQUALS (===)
- OperatorEnum.NOT_EQUALS (!==)
- OperatorEnum.GREATER_THAN (>)
- OperatorEnum.LESS_THAN (<)
- OperatorEnum.GREATER_THAN_OR_EQUAL (>=)
- OperatorEnum.LESS_THAN_OR_EQUAL (<=)
value
: (any
, optional) — The value to compare the field against. Defaults totrue
.
It returns the FusionCollection
instance allows for method chaining.
Example:
// Filter items where "status" is "active"
collection.filter("status", OperatorEnum.EQUALS, "active");
// Filter items where "age" is greater than 18
collection.filter("age", OperatorEnum.GREATER_THAN, 18);
// Filter items where "highlight" is true (using default operator and value)
collection.filter("highlight");
Specifies the field and direction for sorting the collection.
Parameters:
field
(string
): The name of the field to sort by.direction
('asc' | 'desc'
, default'asc'
): The sort direction, either'asc'
for ascending or'desc'
for descending. It returns: TheFusionCollection
instance, allowing for method chaining.
Example:
// orderBy(field: string, direction: 'asc' | 'desc' = 'asc'): FusionCollection
collection.orderBy("date", "desc");
Limits the number of items in the collection to the specified count.
Parameter:
count
(number
): The maximum number of items to include in the collection. It returns: TheFusionCollection
instance, allowing for method chaining.
Example:
// limit(count: number): FusionCollection
collection.limit(3);
Retrieves an array of raw item data from the collection after applying any specified filters, sorting, and limits.
It returns: An array containing the raw data of each item in the collection (any[]
).
Example:
// getItemsArray(): any[]
let itemsArray = collection.getItemsArray();
After applying any specified filters, sorting, and limits, retrieve an array of raw item data from the collection. The difference to getItemsArray()
is that the getMetadataArray()
method excludes the markdown content and includes only the metadata. This method is useful if you need only metadata.
It returns: An array containing the raw data of each item in the collection (any[]
).
Example:
// getMetadataArray(): any[]
let metadataArray = collection.getMetadataArray();
The getOneFilename()
allows you to retrieve one item (FusionItem
and not FusionCollection
) by filename.
Parameter:
filename
(string
) - the filename with the extension and no path (post-1.md
)
It returns: the item (FusionItem
) matching the filename, or null
if no match is found.
Example:
let filename = 'post-1.md';
let dir = './tests/data/no-frontmatter';
let post = new FusionCollection()
.loadFromDir(dir)
.getOneByFilename(filename);
console.log(post.getContent());
console.log(post.getSource()); // 'tests/data/no-frontmatter/post-1.md'
console.log(post.getFilename()); // 'post-1.md'
console.log(post.getName()); // 'post-1'
Retrieves a single item from the collection by its slug.
Parameter:
slug
(string
) - The unique slug identifier for the item to retrieve.
It returns: the item matching the slug, or null
if no match is found. (FusionItem
).
Example:
const item = new FusionCollection()
.loadFromDir('./tests/data/')
.getOneBySlug('post-1');
// So, you can access to some information/content:
item.getContent(); // the markdown content
item.getFields(); // the frontmatter fields
item.getField("title"); // the title frontmatter field
The strong assumption here is that you have the slug
field in your frontmatter sections, for example:
---
title: "Post One"
slug: "post-1"
date: "2023-01-01"
tags: ["typescript", "markdown"]
---
## Hello
Content of the *first post*.
Markdown is widely used for creating structured, lightweight, and readable content. However, Markdown alone lacks the tools for organizing, filtering, and retrieving files based on specific criteria—features that are essential for scaling projects or building complex content management workflows. Fusionable bridges this gap by adding structure and manageability to Markdown collections, so developers can take advantage of the following:
- Metadata-driven Organization: Fusionable reads the frontmatter (YAML metadata) in each Markdown file, making it easy to filter, sort, and group content collections based on properties like tags, categories, publish dates, or any custom fields.
- Content Management as Code: By treating Markdown files as structured data, Fusionable allows developers to use them (their Markdown files) as a content database, with programmatic access for creating dynamic websites, static site generators, or custom documentation tools.
- Enhanced Content Retrieval: Fusionable simplifies accessing Markdown content collections with a fluent API, so you can filter, sort, and retrieve content effortlessly, all in one place.
- Flexibility and Extensibility: Fusionable’s API is built with flexibility in mind. It supports various filtering and sorting operations on Markdown collections, making it ideal for use with JavaScript frameworks or other content pipelines.
If you are interested in contributing to Fusionable, you are welcome! You could evaluate the following:
- Improve the documentation in the readme
- Write a post about Fusionable
- Contributing to the source code, fixing an issue, or implementing a new feature. For that, my suggestion is to read the contributing file
This project was created using
bun init
in bun v1.1.33. Bun is a fast all-in-one JavaScript runtime.