content-tree is a specification that describes the shape of an FT article as an abstract tree. It implements the unist spec. It is intended to be a shared data contract across our CMS (Spark), Content APIs, and Front End systems.
To read the spec, go to SPEC.md. You should read the spec if you are implementing an article renderer, or adding or amending an article component.
The main content-tree specification is defined in a markdown file
SPEC.md: a markdown document defining the specification for the tree. Written in a typescript-like grammar, augmented with a custom[external](#concepts)property modifier.
From this spec, we automatically generate the following as part of the build process:
content-tree.d.ts- Typescript types, which are automatically generated from the markdown spec/schemas- JSON schemas, which are automatically generated from the markdown spec
The Content & Metadata team maintain some Go code to support working with content-tree inside Golang:
content_tree.go- Go structs containing type definitions for the content tree spec to use in Golang applications. Manually updated by Content & Metadata./libraries- Go libraries used to transform the content-tree data structure between other formats
Supporting code:
/tests- tests to validate the schema and transformers/tools- utilities for building and generating the schemas and types
An Abstract Syntax Tree, or AST, is a structured representation of content where the meaning and structure of content are represented as a tree, independent of any particular rendering instruction or representation.
Content Tree is an AST that represents the semantic structure of an article - for example paragraphs, headings, images, other editorial components - without enforcing a particular markup language or visual representation. This makes it easier to use the same content across different products and platforms and contexts.
Node- an element in the tree with atypethat represents a unit of contentParent- aNodewhich haschildrennodesRoot- the single, top-levelNodein a tree
An FT article may contain supplementary assets that are published independently of the content (e.g. images, video clips), or editorial components that pull in data from external systems (e.g. flourish charts, social media posts). When an article is being transmitted over the network (e.g. being fetched via the FT Content API), these external resources are typically referenced by id, and it is up to the consuming application to fetch these resources.
To support both of these use cases, content-tree can exist in different states: a full tree (with resolved external data) and a transit tree (where external resources are referenced by ID). The content and editorial intent remain the same across both states.
To distinguish between transit and full node properties, we use an additional modifier on our typescript properties, called external, which indicates that the property is omitted from the transit representation and must be supplied by the consuming application in order to construct a full tree.
Example: a tweet/X post will be published with an ID, but the actual content of the post should be fetched at render time from the X API.
| spec | transit | full | loose |
interface Tweet extends Node {
id: string
type: "tweet"
external html: string
} |
interface Tweet extends Node {
id: string
type: "tweet"
} |
interface Tweet extends Node {
id: string
type: "tweet"
html: string
} |
interface Tweet extends Node {
id: string
type: "tweet"
html?: string
} |
This package provides typescript types in content-tree.d.ts that can be used to validate the shape of data in a JS/TS application. These types are automatically generated from SPEC.md.
There are three different namespaces exposed, for the different states a tree can be in (see In Transit vs At Rest)
ContentTree.transit- contains only the fields that are published and available from the Content API's responseContentTree.full- contains the full representation of the content, including any data required from external resources (also exposed on the top levelContentTreenamespace)ContentTree.loose- contains the full representation of the content, including any data required from external resources as optional
- Install this repository as a dependency:
npm install https://github.com/Financial-Times/content-tree- Use it in your Typescript / JSDoc code:
import type { ContentTree } from "@financial-times/content-tree"
function makeBigNumber(): ContentTree.BigNumber {
return {
type: "big-number", //will autocomplete in code editor, and be valid
number: "1.2m",
description: "People affected worldwide"
}
}
function makeImageSetNoFixins(): ContentTree.transit.ImageSet {
return {
type: "image-set",
id: "79acd774-6ca7-487d-a257-cbf64d2498d9",
// if you try to add a `picture` here it will get mad
}
}There are also a few JSON schemas generated from the spec.
content-tree.schema.json- JSON schema for the full content-tree shapetransit-tree.schema.json- JSON schema for content-tree excluding anyexternalpropertiesbody-tree.schema.json- JSON schema for the transit tree without theRootnode, containing just the Body definition. This schema defines the data returned in thebodyTreefield of the FT/content-treeAPI.
These libraries are designed for internal use by the Content & Metadata teams, and are used to ensure that all of our Content API representations are aligned with changes in the content-tree spec.
from-bodyxml- converts the legacybodyXMLfield from C&M's internal representation, to a validbodyTreeJSON.to-external-bodyxml- converts content-tree to a stable XML representation, used as a thebodyXMLfield in the FT's/enrichedcontentAPIto-string- converts content-tree to a plain text string
(TODO: how to install / use)
To make a change to the content tree spec:
- Clone this repo and run
npm install - Update SPEC.md with your changes:
- To add a new formatting node, add the definition under the
Formatting Blocks. If it is formatting that can be applied to text in a paragraph, ensure it is added to thePhrasingtype - To add a new storyblock, add the definition under the
Storyblocks. If the block can appear at the top level of the article body, ensure it is also added to theBodyBlocktype definition
- To add a new formatting node, add the definition under the
- Run
npm run buildto updatecontent-tree.d.tsand (if required) theschemasfiles
Once the PR is created, liaise with the Content & Metadata team to ensure the relevant changes are made in the Go libraries and transformers.
For major or non-standard changes, consider creating an issue first, or discussing in the #content-pipeline Slack channel.
(TODO: no good release process yet)
This software is published by the Financial Times under the MIT licence.
Derived from unist © Titus Wormer
