Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Discussion: Using engines in the package.json #286

Open
UlisesGascon opened this issue Oct 16, 2024 · 24 comments · May be fixed by #289
Open

Discussion: Using engines in the package.json #286

UlisesGascon opened this issue Oct 16, 2024 · 24 comments · May be fixed by #289
Assignees
Labels

Comments

@UlisesGascon
Copy link
Member

Currently some packages has drop support to Node.js prior to v18, seems like we didn't upgrade the engines field in the package.json discordantly.

So, I will like to discuss if this a policy that we should follow as an organization to all the packages (so in that case we can also document it).

⚠️ So far seems like Express actually updated the package (ref) but not in others like finalhandler (ref)

Ref:

cc: @expressjs/express-tc @Phillip9587

@UlisesGascon UlisesGascon added meeting discuss top priority Issues which the TC deem our current highest priorities for the project tc agenda labels Oct 16, 2024
@UlisesGascon
Copy link
Member Author

IMO, we should align the engines to the CI versions we test for each release branch. Otherwise, we risk introducing breaking changes without realizing it.

If the package happens to work in older versions of Node.js that we haven't explicitly defined as supported, it should be considered an "unsupported feature." We should encourage end users to use at least the minimum supported version for that release branch. Otherwise, we'll end up "supporting" unintended scenarios based on environments we don't explicitly endorse, leading to issues like unexpected bug reports and slower development due to legacy version compatibility problems.

Additionally, npm is clear in its documentation: unless the user has set the engine-strict config flag, the engines field is advisory only and will simply produce warnings when your package is installed as a dependency. This reinforces the need to clearly define supported versions, ensuring that users are aware of the minimum requirements. (documentation)

@ctcpip
Copy link
Member

ctcpip commented Oct 16, 2024

agree, engines should be added where missing, and should match our CI matrix

@Phillip9587
Copy link

I completely agree with the proposed solution to establish a consistent policy for updating the engines field across all packages. Ensuring that the Node.js version support is clearly defined and aligned accross the packages makes sense from a maintenance and user experience perspective.

I’m happy to contribute and help move this forward by submitting the necessary PRs to update the relevant packages. Let me know how best to coordinate on this!

@ljharb
Copy link
Contributor

ljharb commented Oct 16, 2024

100% that CI should match whatever engines says.

(Changing engines in a way that reduces support is a breaking change, even if the software didn't work on the removed envs, ftr)

@inigomarquinez
Copy link
Member

I completely agree with the proposal!

@bjohansebas
Copy link
Member

I totally agree on keeping the CI in line with what the engines say

@UlisesGascon UlisesGascon linked a pull request Oct 22, 2024 that will close this issue
@UlisesGascon
Copy link
Member Author

Proposal for adr created #289

@ctcpip
Copy link
Member

ctcpip commented Oct 23, 2024

TC agreed today:

  • use engines field
  • engines should reflect minimum supported* node version
    • *supported means what the project has agreed to support, not necessarily the minimum version it can actually run on
  • engines should match CI
    • CI may include older unsupported engines, particularly because it's still useful to know which versions still work and when they break
    • for example, express still runs node 16 and 17 in CI and will continue to do so until a change breaks it
  • ideally, we do not want to arbitrarily drop node version support for projects. in other words, there should not be a major version rev JUST for an engines change. there should be other changes that necessitated the dropping of old node support, e.g. wanting to use new node or ECMAScript features

This is my understanding from today's discussion; others please correct anything if I've gotten it wrong.

@wesleytodd
Copy link
Member

wesleytodd commented Oct 23, 2024

OMG I was so heads down today I totally missed the notification for this. Sorry. I am guessing this one wasn't recorded?

EDIT: I can ask this in slack or find it myself.


use engines field

Did y'all see that post from the fastify folks about this? They had removed it because of the pain it caused. I think we can avoid that as long as we really pick the absolute minimum we support in the major line and dont change it at all for the entire major. But just wanted to call it out in case y'all hand't seen it (I can find a link if necessary nice, the link was in that other pr: fastify/fastify#5667).

in other words, there should not be a major version rev JUST for an engines change. there should be other changes that necessitated the dropping of old node support, e.g. wanting to use new node or ECMAScript features

I am 👍 on this.

@ljharb
Copy link
Contributor

ljharb commented Oct 23, 2024

It's also worth running tests in both "first" and "latest" of a line - iow, if you support ^18, then CI should at least test ~18.0 as well as 18.

@wesleytodd
Copy link
Member

I might have missed it in here, but as we had folks start opening PRs for this in the code and learn yesterday I realized that it wasn't clearly called out that changing the engines field is considered a breaking change. And while we changed it in express if we are going to update them in the sub packages we will be slowly landing those as we land new BREAKING changes. I think that is what y'all meant with " there should be other changes that necessitated the dropping of old node support" but maybe we need to create an ADR with the new format to clarify this?

@UlisesGascon
Copy link
Member Author

Maybe we need to create an ADR with the new format to clarify this?

It is here: #289. I think the major point of the ADR is that the CI should match the engines, and we should include engines where is missing.

So, if we decide to drop support in the CI, it then affects the engines... so that is a major. Adding an engines field to an existing project that aligns with what is supported in the CI is not a major change (IMO) as it essentially reinforces the current state of that version.

That said, I agree that simply bumping Node.js for the sake of bumping Node does not justify a major change, but this might be something to cover in a different ADR related to "when to do a major?" :)

@ljharb
Copy link
Contributor

ljharb commented Oct 27, 2024

@UlisesGascon unfortunately adding an engines field to a project is indeed a breaking change, unrelated to what's in CI, iff the package worked on a now-excluded node version.

@Phillip9587
Copy link

@ljharb I’ve noticed that many packages list in their HISTORY.md that support for Node.js below v18 has been dropped, but the engines field hasn’t been updated to reflect this. Would updating the engines field to match what’s already stated in HISTORY.md be considered a breaking change? Since the documentation already indicates support has shifted, it seems like this change would simply formalize the current compatibility and not necessarily introduce a breaking change. What do you think?

@voxpelli
Copy link

@Phillip9587 Its almost a philosophical question – it’s like bug fixes that are breaking changes but are breaking changes that restore the intended functionality. In the strict way of interpretations they need to be major releases, but some common sense is needed

To me it would make sense to release it as patch – it would be the most correct from the perspective of this at its core being a “bug fix” – ensuring the range is updated in all places.

I believe that @ljharb follows a stricter interpretation than me, and that’s fine, ultimately you need to decide with what you think is the most correct according to SemVer – whether you consider it a bug fix or a new breaking change.

@ljharb
Copy link
Contributor

ljharb commented Oct 27, 2024

@Phillip9587 again, yes - adding engines is only nonbreaking if every engine on which it already works is included in the range.

Indeed “works” can be vague/gray, but despite the impossibility of describing a rubric, I’ve almost never found it a difficult decision to make ad hoc.

@voxpelli
Copy link

@ljharb It will only break when strict engine checks are used and only when installed on engine versions that might anyhow no longer be compatible with the code.

The older engine versions are only safe if the support is rolled back to that of the engine declaration, the current major is marked as deprecated, encouraging them to downgrade or upgrade to a new major with correct claims or the current major gets patched (and the previous releases of it possibly deprecated)

I typically think this should be a major, but in this case I think it’s to be seen as a bug that people upgrading should have already adapted to

@ljharb
Copy link
Contributor

ljharb commented Oct 27, 2024

Many tools check the engines field, and nonzero breakage is breakage, full stop.

I agree that whatever it worked on is what should end up in engines and CI, and then to reduce that, a major bump should occur. Obviously it’s fine to skip the “fix engines/ci to match reality” step as long as engines isn’t released in a non-major.

@bjohansebas bjohansebas removed meeting top priority Issues which the TC deem our current highest priorities for the project tc agenda labels Jan 13, 2025
@ctcpip
Copy link
Member

ctcpip commented Jan 16, 2025

responding to another issue here to keep the conversation centraliized

per what the TC agreed via consensus, "there should not be a major version rev JUST for an engines change".

However, we did not specifically decide anything with regard to this situation which is that we DID drop support, we just forgot to update the engines field. Though there haven't been a lot of cases of it, in practice, we've just updated the engines field without revving major.

I am sympathetic to the view that dropping engines when it would otherwise work with strict engine configurations may be a breaking change. Not one that I consider to be completely unacceptable, but alas...

Ultimately, due to how much of a minor/non-issue it is to leave the engines field alone in these cases, this seems like an easy decision:

  • If a package is completely broken in the older node version(s), then updating engines itself is not a further breaking change -- and therefore acceptable.
  • If a package still works to some extent in the older node version(s) then engines updates need a semver major.

@ljharb
Copy link
Contributor

ljharb commented Jan 16, 2025

I agree with that rubric, although because things can fail to install successfully with an engines change, there might be nonzero breakage that way.

@ctcpip
Copy link
Member

ctcpip commented Jan 16, 2025

Do you mean like, in the case where it's e.g. an unused transitive dependency? So that the fact the library doesn't work at all doesn't matter, thus the breakage via engines is the sole breakage?

@ljharb
Copy link
Contributor

ljharb commented Jan 16, 2025

no, i mean that even if it breaks at runtime in, say, node 16, if it's used inside a try/catch, then a package can still support node 16 with it - but if it adds an engines field that requires 18+, then that's a breaking change for that package and its users.

In other words, you can't ever assume that the engines field is "free" to bump, there's always risk of breakage. which is why it's critically important to always declare it, and when dropping support in a major, always bump it with the major.

@ctcpip
Copy link
Member

ctcpip commented Jan 16, 2025

sure I agree -- but I think the consideration is unchanged. it hinges on the definition of what makes a package "completely broken". I am thinking like, new node module resolution fails, so you can't even run the thing. if I can still pull in a lib as transitive, then it's not "completely broken"

Regardless, I think, in practice, we will just default to not changing in a minor in any case, because it's really low/no value to try to work out whether the change will be breaking or not given these nuances. and like.. who benefits from doing it anyway?

@ljharb
Copy link
Contributor

ljharb commented Jan 16, 2025

Exactly, I agree.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging a pull request may close this issue.

8 participants