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

Use non-nullable types in vscode.d.ts #6907

Closed
jrieken opened this issue May 26, 2016 · 18 comments
Closed

Use non-nullable types in vscode.d.ts #6907

jrieken opened this issue May 26, 2016 · 18 comments
Assignees
Labels
api debt Code quality issues feature-request Request for new features or functionality verification-needed Verification of issue is requested verified Verification succeeded
Milestone

Comments

@jrieken
Copy link
Member

jrieken commented May 26, 2016

TypeScript will introduce support for strict null check and explicit handling of null and undefined - microsoft/TypeScript#7140. This is a great opportunity for us to make the API more explicit and hopefully a chance to prevent programming errors. Parts of the API can move from jsdoc comments into type annotations. For instance, the active editor is declared like this today:

/**
 * The currently active editor or `undefined`. ...
 */
export let activeTextEditor: TextEditor;

and we can move jsdoc type info back into code like so

/**
 * The currently...
 */
export let activeTextEditor: TextEditor | undefined;
@jrieken jrieken self-assigned this May 26, 2016
@jrieken jrieken added feature-request Request for new features or functionality api labels May 26, 2016
@jrieken jrieken added this to the October 2016 milestone Oct 5, 2016
@jrieken jrieken added the debt Code quality issues label Oct 5, 2016
@jrieken
Copy link
Member Author

jrieken commented Oct 10, 2016

depends on TypeStrong/typedoc#234

@jrieken jrieken modified the milestones: November 2016, October 2016 Oct 21, 2016
@jrieken
Copy link
Member Author

jrieken commented Nov 3, 2016

@egamma @dbaeumer Looking for some guidance here. Challenge is that many of the providers define method return types ala Foo | Thenable<Foo>. The current behaviour (spec'd in jsdoc) is that they are allowed to return undefined or null sync or async from with in the promise. That results in the following, very verbose, strict-null-aware signature:

// option 1a, inline
interface CodeLensProvider {
  provideCodeLenses(document: TextDocument, token: CancellationToken): CodeLens[] | undefined | null | Thenable<CodeLens[] | undefined | null>
}

// option 1b, inline but even more verbose
interface CodeLensProvider {
  provideCodeLenses(document: TextDocument, token: CancellationToken): CodeLens[] | undefined | null | Thenable<CodeLens[]> | Thenable<undefined> | Thenable<null>
}

// option 1c, one type for null or undefined
type Nully = null | undefined;

interface CodeLensProvider {
  provideCodeLenses(document: TextDocument, token: CancellationToken): CodeLens[] | Nully | Thenable<CodeLens[] | Nully>
}

// option 2, generic result-type
type Result<T> = T | undefined | null | Thenable<T | undefined | null>

interface CodeLensProvider {
  provideCodeLenses(document: TextDocument, token: CancellationToken): Result<CodeLens[]>
}

// option 3, generic nullable-type
type Nullable<T> = T | undefined | null;    

interface CodeLensProvider {
  provideCodeLenses(document: TextDocument, token: CancellationToken): Nullable<CodeLens[]> | Thenable<Nullable<CodeLens[]>>
}

Options 2 and 3 have the advantage that they can be applied to all providers and make a shorter method signature. On the downside that add another indirection and might make things harder to understand... Tho, also option 1 isn't exactly easy to read and understand. I'd prefer option 1c

@dbaeumer
Copy link
Member

dbaeumer commented Nov 4, 2016

@jrieken hard call: I would actually go for 2.) and name it ProviderResult. Using Nullable makes it bound to nullable. If TS comes up with another feature we might need to change the signature again. With ProviderResult we might be able to simple change that type then.

@bpasero
Copy link
Member

bpasero commented Nov 7, 2016

+1 for option 2 (generic result-type)

@joaomoreno
Copy link
Member

I'd combine 2 and 3, each a different problem in itself, but composable with each other:

type Maybe<T> = T | undefined | null;
type Result<T> = Maybe<T> | Thenable<Maybe<T>>;

interface CodeLensProvider {
  provideCodeLenses(document: TextDocument, token: CancellationToken): Result<T>;
}

@aeschli
Copy link
Contributor

aeschli commented Nov 7, 2016

Wanted to suggest Optional<T> but actually like Maybe<T> even better.
+1 for option 3 with Maybe<T>.
I'd rather not combine it with 2, too many types, and, especially, 'Result' is too general.

@joaomoreno
Copy link
Member

type Eventually<T> = Maybe<T> | Thenable<Maybe<T>>;

😄

@chrmarti
Copy link
Collaborator

chrmarti commented Nov 7, 2016

Keeping the two concepts separate might be easier to read and would allow for more variants of reuse:

type Maybe<T> = T | undefined | null;
type Eventually<T> = T | Thenable<T>;

interface CodeLensProvider {
  provideCodeLenses(document: TextDocument, token: CancellationToken): Eventually<Maybe<CodeLens[]>>;
}

@jrieken
Copy link
Member Author

jrieken commented Nov 7, 2016

Just as a reminder - We aren't just targeting TypeScript-ninjas but also folks that haven't seen a type annotation before.

From my experience, even things like OR-types aren't fully understood/made use of so I am not so sure if a 'double nested generic or-type' is the way to go.

@sandy081
Copy link
Member

sandy081 commented Nov 7, 2016

+1 for option 3

type Nullable<T> = T | undefined | null;

interface CodeLensProvider {
  provideCodeLenses(document: TextDocument, token: CancellationToken): Nullable<CodeLens[]> | Thenable<Nullable<CodeLens[]>>
}

IMO, this makes it API more readable for all kinds of users with out many references. In fact, Nullable is a bit straight so that user do not need to look for its reference.

@chrmarti
Copy link
Collaborator

chrmarti commented Nov 7, 2016

+1 for option 3 then (don't hurt the ninjas!)

@alexdima
Copy link
Member

alexdima commented Nov 8, 2016

+1 for option 2 with a different name for Result, such as ProviderResult.

Reasons:

  • It reduces the burden of reading each and every provider signature, CodeLens[] only appears once
  • It makes that I need to understand only one more concept (i.e. no need to follow through to look up two new concepts Eventually and Maybe)

@jrieken
Copy link
Member Author

jrieken commented Nov 23, 2016

@Microsoft/vscode I am looking for volunteers that migrate their extensions to use strictNull-checks such that we can verify our API

@weinand
Copy link
Contributor

weinand commented Nov 23, 2016

@jrieken how do I migrate my extension?

@jrieken
Copy link
Member Author

jrieken commented Nov 23, 2016

@weinand - to program against the latest API do the following:

  • open your extensions package.json file
  • change the engine.vscode property to *
  • run npm run postinstall (assuming you have the vscode module as dev-dependency)
  • undo the engine-version change from step 2

note tho that this no also depends on TypeScript 2.0

@weinand
Copy link
Contributor

weinand commented Nov 23, 2016

@jrieken done. That was easy.
But I have to set "strictNullChecks": truein my tsconfig.json so that the compiler actually checks for this.

@jrieken jrieken closed this as completed Dec 1, 2016
@jrieken jrieken mentioned this issue Dec 5, 2016
2 tasks
@ramya-rao-a
Copy link
Contributor

@jrieken The test item #16515 is linked here, but the test item does not have anything on non-nullable types.

@jrieken jrieken added feature-request Request for new features or functionality verification-needed Verification of issue is requested and removed feature-request Request for new features or functionality labels Dec 6, 2016
@jrieken
Copy link
Member Author

jrieken commented Dec 6, 2016

@ramya-rao-a This was meant for verification-needed but I have special verifier in mind: those that have already updated their extensions to use strict null checks. I have @mjbvz (ts/js) and @joaomoreno (git exploration) in mind

@mjbvz mjbvz added the verified Verification succeeded label Dec 7, 2016
@egamma egamma mentioned this issue Dec 20, 2016
56 tasks
@vscodebot vscodebot bot locked and limited conversation to collaborators Nov 18, 2017
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
api debt Code quality issues feature-request Request for new features or functionality verification-needed Verification of issue is requested verified Verification succeeded
Projects
None yet
Development

No branches or pull requests