This is the JavaScript track, one of the many tracks on exercism. It holds all the exercises that are currently implemented and available for students to complete. The track consists of various concept exercises which teach the JavaScript syllabus, and various practice exercises, which are unlocked by progressing in the syllabus and can be used to practice concepts learned.
You can find this in the config.json
.
It's not uncommon that people discover incorrect implementations of certain tests, have a suggestion for a track-specific hint to aid the student on the JavaScript specifics, see optimisations in terms of the configurations of jest
, eslint
or other dependencies, report missing edge cases, factual errors, logical errors, and, implement exercises or develop new exercises.
We welcome contributions of all sorts and sizes, from reporting issues to submitting patches, as well as joining the current discussions 💬.
Warning
This guide is slightly outdated and doesn't hold the V3 changes yet.
This guide covers several common scenarios on improving the JavaScript track. There are several other guides about contributing to other parts of the Exercism ecosystem, that are similar to this repository.
- Generic information about track-contributing
- The TypeScript track
- The JavaScript Analyzer
- The TypeScript Analyzer
- The JavaScript Test Runner
Help us keep Exercism welcoming. Please read and abide by the Code of Conduct.
Before contributing code to any existing exercise or any new exercise, please have a thorough look at the current exercises and dive into open issues.
There are two ways to implement new exercises (exercises that don't exist in this track).
- Pick one from the list of exercises (implemented in other tracks).
- Create a new, track-specific exercise from scratch.
The majority of exercises are practice exercises. These exercises are not part of the syllabus (they are not concept exercises), and often have canonical data/shared data between tracks.
Let's say you want to implement a new exercise, from the list of exercises, because you've noticed that this track could benefit from this exercise, really liked it in another track, or just because you find this interesting; the first step is to check for an open issue. If it's there, make sure no one is working on it, and most of all that there is not an open Pull Request towards this exercise.
If there is no such issue, you may open one. The baseline of work is as follows:
- Open a new issue, we'll label it with
new exercise ✨
- We'll assign the issue to you, so you get to work on this exercise
- Create a new folder in
/exercises
- You'll need to sync this folder with the matching config files.
You can use
scripts/sync
to do this. - Create a
<slug>.js
stub file. - Create a
<slug>.spec.js
test file. Here add the tests, per canonical data if possible (more on canonical data below). - Create an
example.js
file. Place a working implementation, assuming it's renamed to<slug>.js
- Create
.meta/tests.toml
. If the exercise that is being implemented has test data in the problem specifications repository, the contents of this file must be a list of UUIDs of the tests that are implemented or not implemented. Scroll down to tools to find configlet which aids in generating this file interactively. - Run the tests locally, using
scripts/test
. - Run the linter locally, using
scripts/lint
. - Create an entry in
config.json
: a unique new UUID (you can use theconfiglet uuid
tool to generate one, scroll down to tools to see how you can get it), give it a difficulty (should be similar to similar exercises), and make sure the order of the file is sane. Currently, the file is ordered first on concept exercise, then on "original core", finally everything else, on difficulty low to high, and ultimately lexicographically. - Format the files, using
scripts/format
.
The final step is opening a Pull Request, with these items all checked off. Make sure the tests run and the linter is happy. It will run automatically on your PR.
If you want to work on a new concept exercise, please talk to the current maintainers of the track.
The steps for a track-specific exercise are similar to those of implementing an established, existing exercise. The differences are:
- You'll have to write a README.md and test-suite from scratch
- You'll have to come up with a unique slug.
- We need to require an icon for it.
- Generate a UUID, for example using configlet.
Open a new issue with your proposal, and we'll make sure all these steps are correctly taken. Don't worry! You're not alone in this.
There are always improvements possible on existing exercises.
For practice exercises, README.md
is generated from a canonical source.
README.md
: the description that shows up on the student's exercise page, when they are ready to start.
It's also downloaded as part of the exercise's data. The README.md
, together with the <slug>.spec.js
file form the contract for the implementation of the exercise.
No test should force a specific implementation, no README.md
explanation should give away a certain implementation.
The README.md
files are generated, which is explained here.
- This file may need to be regenerated to sync with the latest canonical data.
- You may contribute track specific
hints.md
, as listed in that document - You may improve the track-specific
exercise-readme-insert.md
, and regenerate all the READMEs.
For concept exercises, README.md
is generated from the various docs inside the exercise .docs
directory.
introduction.md
: introduce the concept. This is placed on top of the file.instructions.md
: the actual exercise instructions. These follow the introduction.hints.md
: These are hidden behind a button for each task listed ininstructions.md
Only practice exercises require syncing.
Syncing an exercise with canonical data: There is a problem-specifications repository that holds test data in a standardised format. These tests are occasionally fixed, improved, added, removed or otherwise changed. Each change also changes the version of that canonical data. Syncing an exercise consists of:
- updating the
<slug>.spec.js
file; - updating the
.meta/tests.toml
file, if the exercise that is being updated has test data in the problem specifications repository. The contents of this file can be updated using configlet, interactively; - match the
example.js
file to still work with the new tests; and - regenerate the
README.md
, should there be any changes.
Mentor notes are the notes given to the mentors to guide them with mentoring.
These notes do not live in this repository, but instead in the website-copy
repository.
Find their contributing guidelines here.
Some exercises already have automated mentoring support.
These automations don't live in this repository, but instead in the javascript-analyzer
repository.
Find their contributing guidelines here.
There is quite a bit of student-facing documentation, which can be found in the docs
folder.
You may improve these files by making the required changes and opening a new Pull Request.
You'll need LTS or higher Node.js to contribute to the code in this repository.
Run corepack pnpm install
in the root to be able to run the scripts as listed below.
We use the following dependencies:
shelljs
to provide shell interface to scriptseslint
for linting all code in the stub, test file and example filejest
to run all the test files on all example implementationsbabel
to transpile everything so it works regardless of your version of Node.js.
We also use prettier
to format the files.
Prettier is installed when using corepack pnpm install
.
You may use corepack pnpm node scripts/format.mjs
to run prettier.
If you want to auto-format using your editor, install via corepack pnpm install
and it will Just Work™.
If you'd like to download configlet, you can use the fetch-configlet
binary.
It will run on Linux, macOS and Windows, and download configlet
to your local drive.
Find more information about configlet here.
If a track implements an exercise for which test data exists, the exercise must contain a
.meta/tests.toml
file. The goal of thetests.toml
file is to keep track of which tests are implemented by the exercise. Tests in this file are identified by their UUID and each test has a boolean value that indicates if it is implemented by that exercise. Atests.toml
file for a track'stwo-fer
exercise looks like this:
[canonical-tests]
# no name given
"19709124-b82e-4e86-a722-9e5c5ebf3952" = true
# a name given
"3451eebd-123f-4256-b667-7b109affce32" = true
# another name given
"653611c6-be9f-4935-ab42-978e25fe9a10" = false
To make it easy to keep the tests.toml
files up to date, contributors can use the configlet
application's sync
command.
This command will compare the tests specified in the tests.toml
files against the tests that are defined in the exercise's canonical data.
It then interactively gives the maintainer the option to include or exclude test cases that are currently missing, updating the tests.toml
file accordingly.
We have various scripts
to aid with maintaining and contributing to this repository.
/*
* Run this script (from root directory): corepack pnpm node scripts/format.mjs
*
* This runs `prettier` on all applicable files, FORCES using the same version
* as the CI uses to check if the files have been formatted.
*/
Use this action to format all the files using the correct version of prettier.
If you want your editor to do this automatically, install the project development dependencies (corepack pnpm install
), which includes prettier
.
The correct version will be extracted when running .github/workflows/verify-code-formatting.yml
.
/*
* Run this script (from root directory): corepack pnpm node scripts/lint.mjs
*
* This runs `eslint` on all sample solutions (and test) files
*/
If the ASSIGNMENT
environment variable is set, only that exercise is tested.
For example, if you only want to lint two-fer
, you may, depending on your environment use:
ASSIGNMENT=practice/two-fer corepack pnpm node scripts/lint.mjs
Note: on Windows, if you're not in a POSIX style command line, you can use cross-env
to make this work:
# if installed globally
cross-env ASSIGNMENT=practice/two-fer corepack pnpm node scripts/lint.mjs
# otherwise
corepack pnpm dlx cross-env ASSIGNMENT=practice/two-fer node scripts/lint.mjs
/**
* Run this script (from root directory): corepack pnpm node scripts/test.mjs
*
* This runs `jest` tests for all sample solutions
*/
If the ASSIGNMENT
environment variable is set, only that exercise is tested.
For example, if you only want to test the example.js
for two-fer
, you may, depending on your environment, use:
ASSIGNMENT=practice/two-fer corepack pnpm node scripts/test.mjs
Note: on Windows, if you're not in a POSIX style command line, you can use cross-env
to make this work:
# if installed globally
cross-env ASSIGNMENT=practice/two-fer corepack pnpm node scripts/test.mjs
# otherwise
corepack pnpm dlx cross-env ASSIGNMENT=practice/two-fer node scripts/test.mjs
/**
* Run this script (from root directory): corepack pnpm node scripts/sync.mjs
*
* This script is used to propagate any change to root package.json to
* all exercises and keep them in sync.
* There is a CI step which checks that package.json in root & exercises match
* (see checksum script for more info).
*/
If the ASSIGNMENT
environment variable is set, only that exercise is tested.
For example, if you only want to sync the files for two-fer
, you may, depending on your environment, use:
ASSIGNMENT=practice/two-fer corepack pnpm node scripts/sync.mjs
Note: on Windows, if you're not in a POSIX style command line, you can use cross-env
to make this work:
# if installed globally
cross-env ASSIGNMENT=practice/two-fer corepack pnpm node scripts/sync.mjs
# otherwise
corepack pnpm dlx cross-env ASSIGNMENT=practice/two-fer node scripts/sync.mjs
/*
* Run this script (from root directory): corepack pnpm node scripts/checksum.mjs
*
* This will check root `package.json` matches each exercise's `package.json`.
* But the catch is there are some dependencies that are only used at build-time and not served to end-users
* We skip those dependencies while performing the checksum.
* See `SKIP_PACKAGES_FOR_CHECKSUM` in helpers.js for a list of skipped packages.
*/
/**
* Run this script (from root directory): corepack pnpm node scripts/ci-check.mjs
*
* This will run the following checks:
*
* 1. Check config in all exercises matches
* 2. Checks stubs exist
* 3. Run eslint to check code-style
*/
Run this script to check stubs, configuration integrity and lint the code.
/**
* Run this script (from root directory): corepack pnpm node scripts/ci.mjs
*
* This will run the following checks:
*
* 1. Find the exercises
* 2. Run tests against sample solutions
*/
Run this script to test all exercises.
/**
* Run this script (from root directory): corepack pnpm node scripts/name-check.mjs
*
* This will run the following checks:
*
* 1. Package name is of the format "@exercism/javascript-<exercise>"
*
* This script also allows fixing these names: corepack pnpm node scripts/name-check.mjs --fix
*/
Run this script to check if the package name in package.json of exercises is in the expected format or to fix it.
/**
* Run this script (from root directory): corepack pnpm node scripts/name-uniq.mjs
*
* This will run the following checks:
*
* 1. All exercises have unique package names in their package.json files.
*/
Run this script to check if there is any duplicate package name.
/**
* Run this script (from root directory): corepack pnpm node scripts/directory-check.mjs
*
* This will run the following checks:
*
* 1. The package has the correct directory based on the path to the exercise.
*
* This script also allows fixing these directories: corepack pnpm node scripts/directory-check.mjs --fix
*/
Run this script to check if the package repository directory in package.json of the exercises is in expected format or to fix it.
If the ASSIGNMENT
environment variable is set, only that exercise is tested.
For example, if you only want to test the directory for concept/closures
, you may, depending on your environment, use:
ASSIGNMENT=concept/closures corepack pnpm node scripts/directory-check.mjs
Note: on Windows, if you're not in a POSIX style command line, you can use cross-env
to make this work:
# if installed globally
cross-env ASSIGNMENT=concept/closures corepack pnpm node scripts/directory-check.mjs
# otherwise
corepack pnpm dlx cross-env ASSIGNMENT=concept/closures node scripts/directory-check.mjs