Observable Plot is open source and released under the ISC license. You are welcome to send us pull requests to contribute bug fixes or new features. We also invite you to participate in issues and discussions. We use issues to track and diagnose bugs, as well as to debate and design enhancements to Plot. Discussions are intended for you to ask for help using Plot, or to share something cool you’ve built with Plot. You can also ask for help on the Observable Forum and the Observable community Slack.
We request that you abide by our code of conduct when contributing and participating in discussions.
To contribute to Observable Plot, you’ll need a local development environment to make and test changes to Plot’s source code. To get started, follow GitHub’s tutorial on forking (and cloning) a repository. Once you’ve cloned your fork of the Plot repository, open a terminal and cd
in your forked repository. Then run Yarn to install dependencies:
yarn
You may encounter an error installing node-canvas, such as:
node-pre-gyp ERR! install response status 404 Not Found on https://github.com/Automattic/node-canvas/releases/download/v2.9.1/canvas-v2.9.1-node-v93-darwin-unknown-arm64.tar.gz
node-pre-gyp WARN Pre-built binaries not installable for [email protected] and [email protected] (node-v93 ABI, unknown) (falling back to source compile with node-gyp)
node-pre-gyp WARN Hit error response status 404 Not Found on https://github.com/Automattic/node-canvas/releases/download/v2.9.1/canvas-v2.9.1-node-v93-darwin-unknown-arm64.tar.gz
If this happens, you will need to compile node-canvas from source. On macOS you can use Homebrew to install the needed dependencies:
brew install pkg-config cairo pango libpng jpeg giflib librsvg
After making changes to Plot’s source code, run Plot’s test suite to verify that your code is doing what you expect and that you haven’t introduced any other unexpected changes in behavior. Plot has two types of tests: unit tests and snapshot tests. Tests are run automatically on pull requests (via GitHub Actions), but you’ll want to run them locally to verify your changes before opening a pull request. To run the tests, use Yarn:
yarn test
This will also run ESLint on Plot’s source to help catch simple mistakes, such as unused imports.
Please run Prettier before submitting any pull request. Check “format on save” in your code editor, or run:
yarn prettier --write .
A test coverage report can be generated with c8, in text and lcov formats, to help you identify which lines of code are not (yet!) covered by tests. Just run:
yarn test:coverage
Unit tests live in test
and have the -test.js
file extension; see test/marks/area-test.js
for example. Generally speaking, unit tests make specific, low-level assertions about the behavior of Plot’s API, including internals and helper methods. If you add a new feature, or change the behavior of an existing feature, please update the unit tests so that we can more easily maintain your contribution into the future. For example, here’s a unit test that tests how Plot formats months:
it("formatMonth(locale, format) does the right thing", () => {
assert.strictEqual(Plot.formatMonth("en", "long")(0), "January");
assert.strictEqual(Plot.formatMonth("en", "short")(0), "Jan");
assert.strictEqual(Plot.formatMonth("en", "narrow")(0), "J");
});
Plot’s unit tests are written with Mocha.
If you like, you can also run Mocha in watch mode for a specific file, so that unit tests re-run automatically when you make changes. For example:
yarn run mocha --conditions=mocha --parallel --watch test/marks/bar-test.js
Snapshot tests live in test/plots
and are registered in test/plots/index.ts
; see test/plots/aapl-bollinger.ts
for example. Unlike unit tests which only test individual methods, snapshot tests actually visualize data—they’re more representative of how we expect people will use Plot. Snapshot tests can also serve as examples of how to use the Plot API, though note that some of the examples intentionally test edge case of the API and may not embody best practices. Each snapshot test defines a plot by exporting a default async function. For example, here’s a line chart using BLS unemployment data:
import * as Plot from "@observablehq/plot";
import * as d3 from "d3";
export async function lineUnemployment() {
const bls = await d3.csv<any>("data/bls-metro-unemployment.csv", d3.autoType);
return Plot.plot({
marks: [
Plot.ruleY([0]),
Plot.lineY(bls, {x: "date", y: "unemployment", z: "division"})
]
});
}
When a snapshot test is run, its output is compared against the SVG or HTML snapshot saved in the test/output
folder. This makes it easier to review the effect of code changes and to catch unintended changes. Snapshot tests must have deterministic, reproducible behavior; they should not depend on live data, external servers, the current time, the weather, etc. To use randomness in a test, use a seeded random number generator such as d3.randomLcg.
To add a new snapshot test, create a new JavaScript file in the test/plots
folder using the pattern shown above. Then export your snapshot test function from test/plots/index.ts
. For example:
export * from "./moby-dick.ts";
The best thing about snapshot tests is that you can see the live result in your browser as you make changes to Plot’s source code! This lets you immediately assess visually what Plot is doing. To preview snapshot tests during development, Plot uses Vite. To start Vite:
yarn dev
This will open http://localhost:8008/ in your browser where you can choose a snapshot test. As you edit the source, the current test will update live in your browser as you save changes. You can change the selected test from the drop-down menu. When the drop-down menu is focused, the left and right arrow keys cycle between tests.
When previewing snapshot tests, consider using your browser’s debugger or element inspector to assist development.
Running Plot’s snapshot tests will automatically generate any missing snapshots in test/output
. You should git add
these before committing your changes. (If you forget, your PR will fail in CI, and you’ll get a reminder.) Changed snapshots are saved alongside the originals with a -changed
suffix for visual inspection. If your code intentionally changes some of the existing snapshots, simply blow away the existing snapshots and run the tests again. You can then review what’s changed using git diff
.
rm -rf test/output
yarn test
When submitting a pull request, please remember to update Plot’s documentation to reflect changes to the public API. You are also welcome to edit Plot’s CHANGELOG.md to assist with writing future release notes. In addition, please reference any related issues (or discussions) in your pull request description.
If you’d like to share a live demonstration or motivating example of your change to Plot, you can regenerate Plot’s release bundle using Yarn:
yarn prepublishOnly
The generated bundle dist/plot.umd.js
can then be loaded like so:
<script src="https://cdn.jsdelivr.net/npm/d3@7"></script>
<script src="plot.umd.js"></script>
Alternatively, you can attach the dist/plot.umd.js
file to an Observable notebook, and then load it like so:
Plot = require(await FileAttachment("plot.umd.js").url())