question-mark
Stuck on an issue?

Lightrun Answers was designed to reduce the constant googling that comes with debugging 3rd party libraries. It collects links to all the places you might be looking at while hunting down a tough bug.

And, if you’re still stuck at the end, we’re happy to hop on a call to see how we can help out.

Automated Migration for Breaking Changes to the Type System

See original GitHub issue

Automated Migration for Breaking Changes to the Type System

Search Terms

codemod, migration, upgrade

Suggestion

semi-automated migration when there are breaking changes in the type system

One way of implementing this would be via codemods (could use the TS Compiler API) that do trivial update tasks as described in https://github.com/microsoft/TypeScript/issues/33272

Use Cases

  • Would make it easier for users to upgrade to the latest TypeScript
  • Would make it easier for the TS team to upgrade TypeScript
  • Might enable TS to evolve faster, as the pain from these “breaking” changes will be smaller

The changes described in https://github.com/microsoft/TypeScript/issues/33272, for example, were mostly trivial, and ideally would not need much human intervention.

Type-only changes are especially amenable to this kind of treatment, because no one’s app will misbehave at runtime if there is a bug in the transformation.

I say ‘semi-automated’ because in cases where the tool can’t figure out the problem, the tool could yield to human judgment. Tooling for this doesn’t have to be perfect, but a small amount of automation could go a long way.

Examples

example.ts

const myPromise: Promise<{}> = dontCarePromise();

tsc --upgrade --from=3.5 --to=3.6 example.ts

const myPromise: Promise<unknown> = dontCarePromise();

The example is from https://github.com/microsoft/TypeScript/issues/33272.

In a case where the conversion would lead to a new type error, I’d expect the tool to bail out, explain the situation, and let the human decide what to do.

Checklist

My suggestion meets these guidelines:

  • This wouldn’t be a breaking change in existing TypeScript/JavaScript code
  • This wouldn’t change the runtime behavior of existing JavaScript code
  • This could be implemented without emitting different JS based on the types of the expressions
  • This isn’t a runtime feature (e.g. library functionality, non-ECMAScript syntax with JavaScript output, etc.)
  • This feature would agree with the rest of TypeScript’s Design Goals.

Issue Analytics

  • State:open
  • Created 4 years ago
  • Reactions:11
  • Comments:7 (6 by maintainers)

github_iconTop GitHub Comments

2reactions
RyanCavanaughcommented, Sep 11, 2019

Broadly speaking there are two distinct categories of things to consider here:

  • Breaks for which the root cause is tractably identifiable by the type system (Type I)
  • Breaks for which the root cause is not tractably identifiable by the type system (Type II)

An example Type I break would be strictClassPropertyInitialization - by the construction of the error itself, it’s trivial for TS to identify that “I am issuing an error because of a rule that was added in a breaking manner”.

An example Type II break would be the Set constructor change identified in #33272 - this was caused by a subtle change in inference rules, and the exact error occurring later in the code isn’t something TS could plausibly trace back to a change in behavior.

We can create Quick Fixes for Type I errors, and typically do. These also tend to be very well-documented and well-thought-out changes because they’re typically the “We are breaking you on purpose” kind of breaking change. Many Type I breaks are also associated with a commandline flag to turn them off.

For a Type II error, even if we could figure out what happened, often there’s no “quick fix” because it depends on what the code was trying to do in the first place, and the change that does need to occur might be very lexically far away from where the first actual type error is issued.

We can provide tooling all day long for Type I breaks, but I would argue that this is a bit of a low-value activity. These breaks are often not particularly painful, are signaled well in advance, and usually have a simple commandline fix or can be bulk-addressed by a sufficiently clever RegExp. Adopting something like Facebook’s codemod tool for offering a CLI experience for these fixes is something we might look in to.

Type II breaks are vexing. Often we don’t even really know we’re making them - the filter(Boolean) break again from #33272 is an example. And even if we did fully realize this was going to be a break, it’s not clear what we’d do about it. There’s no obvious quickfix to issue for “This expression is going to change to an arguably more-accurate type in a way that might break downstream code that makes certain other assumptions”.

For example, let’s say you wrote some code

const arr = [1, 2, 3, "four", "five"];
const arr2 = arr.filter(x => +x === x);
arr2.push("six");

Maybe in the future, TS gets smarter and says +x === x is a type-guarding expression for x is number, and also gets smarter by identifying that array.filter(type guard function) returns a more-specific array type. At first glance, this isn’t a breaking change… but it is: the arr2.push("ok") line is now an error because arr.filter returned number[] instead of (string | number)[].

In even the most trivial example, the error is very causally disconnected from the original change, and it’s not even clear what the right “fix” to issue is.

What it seems like you want for Type II errors is just a tool that runs tsc, takes the errors, and inserts either assertions or // @ts-ignore comments depending. For example, I’d probably want one of two outputs depending on error density:

const arr = [1, 2, 3, "four", "five"];
const arr2 = arr.filter(x => +x === x);
// @ts-ignore Error 'cannot convert "six" to 'number'" introduced in TS 4.2
arr2.push("six");

vs

const arr = [1, 2, 3, "four", "five"];
// TODO: Multiple errors involving 'arr2' introduced in TS 4.2
const arr2: any = arr.filter(x => +x === x);
arr2.push("six");
arr2.push("seven");
arr2.push("eight");
1reaction
ortacommented, Sep 10, 2019

This is cool, I’d also like to see the same on a per-flag basis, e.g. tsc --upgrade-flag strictFunctionTypes example.ts

Read more comments on GitHub >

github_iconTop Results From Across the Web

Automated Migration in Entity Framework 6
Entity Framework introduced automated migration so that you don't have to process database migration manually for each change you make in your domain...
Read more >
Breaking changes in .NET 7
If you're migrating an app to .NET 7, the breaking changes listed here might affect you. Changes are grouped by technology area, such...
Read more >
Breaking Changes · microsoft/TypeScript Wiki
TypeScript is a superset of JavaScript that compiles to clean JavaScript output. - Breaking Changes · microsoft/TypeScript Wiki.
Read more >
Move fast and migrate things: how we automated ...
Additive changes like adding a column were made in the pre-deploy migrations (before the model code that needed it was deployed). Destructive ...
Read more >
Migrating to 8.0 | Elasticsearch Guide [8.5]
Breaking changes edit. The following changes in Elasticsearch 8.0 might affect your applications and prevent them from operating normally.
Read more >

github_iconTop Related Medium Post

No results found

github_iconTop Related StackOverflow Question

No results found

github_iconTroubleshoot Live Code

Lightrun enables developers to add logs, metrics and snapshots to live code - no restarts or redeploys required.
Start Free

github_iconTop Related Reddit Thread

No results found

github_iconTop Related Hackernoon Post

No results found

github_iconTop Related Tweet

No results found

github_iconTop Related Dev.to Post

No results found

github_iconTop Related Hashnode Post

No results found