How to choose dependencies β
An IT project will have graphical interfaces β AKAΒ§ frontends: web/mobile apps β, backends, databases, potentially in different languages. Which means we start this game with already our share of major upgrades:
- Language upgrades
- Database upgrades
- OS upgrades (my linux server from 18.04 to 20.04 then 22.04, β¦)
- Provider plan changes β AWS, CF Workers, Google Maps or Mapbox, Mixpanel, etc.
- Development tool upgrades: vite/rollup/TypeScript/Capacitor/Netlify/CloudflareWorkers/GCP/β¦
That already represents a sure amount of maintenance work, doesnβt it?
My main goal now is to diminish the number of other major β breaking β upgrades I will encounter. For that I have a few strategies.
What the dep? β Red flags π© β
π© 1. diminish as much as possible the number of dependencies β
It seems obvious, but I still see some projects where object-assign
or pad-left
are installed.
π© 2. Avoid framework satellites β
vue-*, react-*, express-*, you get the drill.
Routing or i18n do not require Vue nor React.
Parsing a request body does not require Express nor Fastify.
I could go on for a while.
Any library like this you install will couple you further to the parent framework: The parent framework has a major version release? Your satellite library will have one too. 2+ major upgrades for the price of 1, yey π
There is no good reason to install a framework satellite, hiding behind your laziness the ease of use will not help.
Bottom line: favor framework-agnostic libraries, best-in-class are those with adapters like fullcalendar, body-parser, i18next/FormatJS/Fluent, floating-ui, etcβ¦.
π© 3. Avoid libraries bringing their own standards β
fp-ts
, effect
, ramda
, Angular
, NestJS
, etcβ¦ Using such libraries makes you completely tied to them by giving capabilities other libs donβt. That is their key to keep you in ; Thereβs no going back.
Abstractions are good but favor your own, or smaller ones. Unless you facade everything, creating your own std library with a patchwork of tools, why not.
Yellow flags β οΈ β
β οΈ 1. Native APIs duplicates β
The web platform is improving super fast, making some (parts of) libraries stale: lodash, underscore, MoutJS, Ramda, BigJS (-> bigint), β¦
To fill web platform gaps, favor polyfills if possible (see polyfill.io), smaller libraries otherwise (like just-* collection).
β οΈ 2. Lots of major versions in short amount of time β
Check out the npm versions
tab. Why: I don't want to be dependent on someone changing their mind. React I look at you, but not only.
Letβs compare:
React has released 4 major versions in 9 years (completely changing paradigms twice).
Anguar has released 18 major versions in 9 years, including Angular v1 -> v2
Vue has released 3 major versions in 9 years.
Svelte has released 5 major versions in 8 years (changing completely paradigms for v5).
Solid has released 1 major version in 3 years.
i18next has released 24 major versions in 11 years.
@formatjs/intl has released 3 major versions in 4 years.
This gives you an idea of the pace.
These numbers should only serve as warnings, make the decision by reading Migration Guides and/or changelogs..
- Breaking changes are overall anecdotic -> β .
- Breaking changes include paradigms shifts once -> β οΈ, check for how long the last shift has lasted.
- Breaking changes include paradigms shifts more than once -> big β οΈ.
- No changelog nor migration guide -> π©.
β οΈ 3. Libraries encouraging hard-to-test code practices β
For instance: tanstack query or form, π. When I look at their usage, thereβs no way to test anything outside the framework world ; Big π‘ flag.
The green flags β β libraries I am fine using β
Libraries which are framework-agnostic and have a small API surface. For example:
- fuse.js (fzf-js, β¦): I would encourage to add them to an
array
std, however the API is simple. If I ever need to migrate that, it will take me less than a day. Especially if part of my std (facaded). - floating-ui
- is-email
- just-*, just-pipe, just-omit, just-pick. Facaded in an std.
- qs-esm
Final word: we will need frameworks & libraries β
That being said, we cannot avoid installing some of those. However, we can choose framework-free dependencies and to facade our frameworks. This is a pre-requisite to be able to use frameworks in a decoupled way.
Which leads to β¦
Golden Rules β
I want to introduce two golden rules:
- Never mention a technology when defining concepts
- Always mention the technology when implementing concepts
A Cache
example β
// library/cache/definition.ts
export interface Cache<T> {
get(key: string): T
set(key: string, value: T): void
}
// library/cache/redis.ts
export function makeRedisCache<T>({ connectionString, β¦ }): Cache<T> { β¦ }
A NaturalSearchEngine
example β
// library/natural-search-engine/definition.ts
export interface NaturalSearchEngine<T> {
feed(documents: T[]): Promise<void>
search(query: string): Promise<SearchResult<T>>
}
interface SearchResult<T> {
score: number
value: T
}
// library/natural-search-engine/typesense.ts
export function makeTypeSenseNaturalSearchEngine<T>({ url, β¦ }): NaturalSearchEngine<T> {}