This template project is part of Sourdough, a set of JavaScript templates that were originally developed at Northeastern for their Software Engineering class in spring of 2026.
The functional content of this project is a minimal "transcript service" that
registers students by assigning them an ID and then lets course grades be
added and queried. Everything in the ./src directory can be deleted to
create a true empty project.
The base project configuration follows a philosophy of "minimalism, mostly." Project configuration should be minimal and have a bias towards implicit defaults. Deviations from this principle should be justified and documented (here or elsewhere).
Notable exceptions to this principle:
-
.gitignoretakes a kitchen-sink approach and should freely accept additions. (For example, if a student accidentally checks in a file that could have been ignored, it makes sense to add that file here.) -
The ESLint configuration is a maximalist attempt at keeping new TypeScript programmers on the rails in a complicated codebase, and also giving them a sense of working inside style conventions of a project that may differ from their own.
-
If we can have all project variants using the exact same
tsconfig.jsonfile oreslint.config.mjsfile by adding a bit of cruft to the base configuration, that's a reasonable trade. Things are going to inevitably get copy-pasted, and so the fewer copies of configuration files there are, the better.This is why the
.vscode/settings.jsonapplies Prettier to html and css files even though that's not relevant to the base project, and whyeslint.config.mjsincludes React's Rules of Hooks despite most of the project variants not including any React code.
This sets up a set of commands that CS4530 templates should consistently support:
npm run checkruns TypeScriptnpm run lintruns ESLint, andnpm run lint:fixruns eslint with the--fixoptionnpm run prettierchecks formatting, andnpm run prettier:fixwrites formatted files back to disknpm run testruns Vitest tests and reports coverage
When appropriate, projects should also have the following scripts:
npm run devstarts a development server or watch processnpm run buildprepares the project for production-style deploymentnpm startruns the project in production style
These are tested by github actions in .github/workflows/main.yml.
This base project has an opinionated ESLint configuration that relies on typed linting. The ESLint configuration makes some assumptions about project structure:
- Frontend code is code that lives in
./frontendor./client, and optionally uses React and JSX. This code is subject to different linter rules. - Test code lives in a
**/testsdirectory OR has a*.spec.ts(x)or a*.test.ts(x)filename. Tests can use devDependencies, unlike other code. - Config files all have
*.config.mjsfilenames (vite, vitest, playwright, and eslint all can follow this convention). These can also import devDependencies, unlike other code. This means we're not using TypeScript to check our config files. - Most everything should be registered as
error. Warnings don't fail CI checks. Exceptions should have a documented reason. Notable exceptions:no-consoleiswarnbecause no-console regularly gets turned off by line or file specific rules: we want to discourage excessiveno-consoleuse but it is more like the admonition to not check in commented-out code: it's mostly a problem when done excessively and it's easy to check in visual inspection.prettieriswarnbecause red squigglies forprettierare especially distracting and we can check for prettier failures in CI separately.- We do not override the default setting of
warnforreact-hooks/exhaustive-depsin the default configuration. This rule makes the (horrible) suggestion to remove the dependency array, and people breaking their projects by blindly following that suggestion would be a bad outcome.
TypeScript is configured with options that support
type stripping.
Beyond this, the TypeScript configuration enables
noFallthroughCasesInSwitch, noImplicitOverride, noImplicitReturns, and
noUncheckedIndexAccess, which are linter-like properties that don't seem to
be well-supported by typed linting in ESLint.
Matt Pocock's cheat sheet is a reasonable source for more on minimal typescript configuration.
The .prettierrc file is intended to use some reasonable defaults. A
.vscode/settings.json file is added to encourage Visual Studio Code to treat
Prettier as the default formatter for javascript, typescript, json, css, and
html files even if a students' global configuration uses other defaults.
The .prettierrc, .gitattributes, and .vscode/settings.json files
conspire to generally force projects to use \n file endings instead of
Windows-style \r\n line endings (LF instead of CRLF).
Different repositories live in different branches:
Base configuration:
`base`
|
v add an Express server and API tests
`express`
|
v add a Vite frontend (+ Playwright tests) for a simple client/server setup
`fullstack`
|
v add React to the frontend
`fullstack-react`