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

Add smarter type inference for ofType pipe. #1183

Merged
merged 1 commit into from
Oct 16, 2018

Conversation

rkirov
Copy link
Contributor

@rkirov rkirov commented Jul 12, 2018

It is based on TS2.8 introduced conditional types. Upgrade package.json
to that version.

Also remove the deprecated non-pipable version of ofType so that we
don't have to keep complicated types in two places.

Fixes some tests post removal of non-pipable version.

Original implementation by @mtaran-google.

Fixes #860

@coveralls
Copy link

coveralls commented Jul 12, 2018

Coverage Status

Coverage decreased (-0.006%) to 88.394% when pulling 200cb74 on rkirov:oftype_types into 41a0d45 on ngrx:master.

@rkirov rkirov force-pushed the oftype_types branch 4 times, most recently from 35b3ab4 to d04cae3 Compare July 13, 2018 00:58
import { filter } from 'rxjs/operators';

// Removes types from T that are not assignable to U.
export type Filter<T, U> = T extends U ? T : never;

@Injectable()
export class Actions<V = Action> extends Observable<V> {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wonder if I should also remove lift and ctor impl as proposed in #860?

@@ -0,0 +1,12 @@
export default {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I copied those from testing but I have not idea whether they are correct in the context of compat.

@timdeschryver
Copy link
Member

timdeschryver commented Jul 24, 2018

Should we allow an "infite" number of ofType arguments?

@mtaran-google
Copy link
Contributor

That would be ideal, but the type system in TS 2.9 isn't powerful enough to check that many. 3.0 (in RC now) improves on this but I'm not sure if it'll be enough. It'll be interesting to check out when it lands.

@timdeschryver
Copy link
Member

timdeschryver commented Jul 25, 2018

@mtaran-google I see, thanks.

Just one more thing, I thought the following would give me compile errors but they don't:

// this is OK
ofType<Search>(BookActionTypes.Search),

// I would expect an error here
ofType<Search>('sss'),

// I would expect an error here, because the payload of `SearchComplete` is an array
ofType<Search>(BookActionTypes.SearchComplete),
map(action => action.payload),
  switchMap(query => {
    if (query === '') {
        return empty();
     }

I fiddled with it in the past and came up with this which is giving compile errors.

@rkirov
Copy link
Contributor Author

rkirov commented Jul 25, 2018

@timdeschryver That's a good point. Let's recap the desired outcomes:

  1. ofType('x') to just work based in inference
  2. ofType<SomeType>('someActionType') to keep working as is.
  3. ofType<SomeType>('typo') to error if SomeType.type != 'typo' (both of your examples boil down to this).

My current PR has 2) but not 3), and yours I think has 3) but not 2), because you have two free type variables export function ofType<V extends Action, Type extends string> and no defaults. Maybe I am missing something, but I don't think you can have 2) and 3) at the same time.

I think 3) is not as bad as it seems. If we have property 1) eventually, code will be written without any explicit types in <>, thus irrelevant how is the argument vs explicit type annotation resolved. Until then one just has to use the rule-of-thumb "if <> explicitly specified, it wins", i.e. think of it as a type assertion.

@rkirov
Copy link
Contributor Author

rkirov commented Jul 25, 2018

However, based on your your example it looks like the T1,T2,..., etc expansions are not necessary. I think the following works the same.

export function ofType<
  V extends Extract<U, { type: T }>,
  T extends string = string,
  U extends Action = Action
>(...t: T[]): OperatorFunction<U, V>;

Also one final thing, my work here is prompted by attempting to turn on --strictFunctionTypes and hitting a lot of ngrx issues. Please do your testing with that flag on. I think any solution that doesn't have at least two parameters U -> V to capture the broader observable input and the narrowed out observable output, will not work with --strictFunctionTypes.

@alex-okrushko
Copy link
Member

Extract showed up in 2.8, while ngrx is still at 2.7

@rkirov
Copy link
Contributor Author

rkirov commented Jul 25, 2018

@alex-okrushko That is the blocker for all this. TS2.8 is required for conditional types, Maksym's original change a custom alias for Extract.

@timdeschryver
Copy link
Member

@rkirov Yep you're completely right.
The solution I came up with doesn't support ofType<SomeType>('someActionType').

And the snippet you posted has the same outcome.

export function ofType<
  V extends Extract<U, { type: T }>,
  T extends string = string,
  U extends Action = Action
>(...t: T[]): OperatorFunction<U, V>;

@rkirov
Copy link
Contributor Author

rkirov commented Jul 26, 2018

Ok, I just discovered one more issue that might be the deal breaker here. A lot of teams do not pass they type argument on their actions observable is it just of type - Actions. Now, if that property is Actions<AllPossibleActions> this change is great.

However, for unqualifed Actions this change (in both proposed forms) is actually a regression. That means if one has private actions: Actions; then this.actions.pipe(ofType('something')) creates Observable<never> instead of at least Observable<Action>.

This is expected because Extract<{type: string}, {type: 'a'}> is never and not {type: 'a'}. In a more general form Extract<string, 'a'> is never though one can expect it to be 'a'. Not sure if TS has plans to change that.

@rkirov
Copy link
Contributor Author

rkirov commented Aug 1, 2018

I updated this PR with what is my final proposal for the future types of 'ofType'. Added comments that should capture the upside and downsides of that approach. Hopefully, that would make the type soup a bit more maintainable.

Internally, I appears we have an agreement that this is good and are going to change it with a patch before this PR lands here. Here it appears to be stuck behind angular, bazel, typescript versioning hell, that I can't resolve.

@MikeRyanDev and @brandonroberts please at least verify that there are no major concerns with this proposal, otherwise we would end up forking out internal google users from the mainstream branch here.

@MikeRyanDev
Copy link
Member

@rkirov This proposed change looks good to me.

@brandonroberts
Copy link
Member

@rkirov I agree also

import { filter } from 'rxjs/operators';

// Removes types from T that are not assignable to U.
export type Filter<T, U> = T extends U ? T : never;
Copy link
Member

@timdeschryver timdeschryver Aug 1, 2018

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Because we're using Extract this can be removed now.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Indeed.

modules/effects/spec/actions.spec.ts Show resolved Hide resolved
@@ -19,17 +22,79 @@ export class Actions<V = Action> extends Observable<V> {
observable.operator = operator;
return observable;
}

ofType<V2 extends V = V>(...allowedTypes: string[]): Actions<V2> {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This may sound stupid, but here it goes anyways:
Since we have the fallback ofType, could we not remove this but mark it as deprecated?
With the next release we can remove it.
I say this because while the compat solution works, the user would have to change the import statement?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have no opinions on backwards compat (google operates on one-version policy), and will do whatever you guys decide.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was planning on doing a 6.1 release, so maybe we can deprecate the instance operators before release and then move forward with the removal for the next major release.

package.json Outdated
@@ -143,7 +143,7 @@
"tslib": "1.6.0",
"tslint": "^5.0.0",
"tsutils": "2.20.0",
"typescript": "~2.7.2",
"typescript": "~2.8.4",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this means that angular version would need a bump as well:

@angular/[email protected] requires typescript@'>=2.7.0 <2.8.0' but 2.8.4 was found instead.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Indeed, at this point this PR is doing too much. Another PR is needed to get to TS 2.8, with all the complications involved around angular, bazel, router(?). I can't do that upgrade, but will rebase this PR once that happens.

@rkirov rkirov mentioned this pull request Aug 1, 2018
@rkirov
Copy link
Contributor Author

rkirov commented Aug 1, 2018

Opened #1220 to track the compiler upgrade. This PR is blocked (will respond to review comments though) until that gets resolved.

@rkirov
Copy link
Contributor Author

rkirov commented Aug 15, 2018

rebased the PR, will take a look at the CI results tomorrow.

@rkirov
Copy link
Contributor Author

rkirov commented Aug 15, 2018

CI is green, please proceed with a regular review. I don't fully understand the compat strategy, mostly copy/pasted from other directories.

@timdeschryver
Copy link
Member

I think the compat version won't be needed. The store.ofType (and store.select) operators are now marked as deprecated and will be removed the next version.
Before you change anything, wait untill @brandonroberts 's confirmation tho.

@brandonroberts
Copy link
Member

@rkirov @timdeschryver I think its safe to remove the compat package also. The commit message does need to reflect the breaking change of removing the instance operator.

@@ -11,12 +11,20 @@ import { hot, cold } from 'jasmine-marbles';
import { of } from 'rxjs';

describe('Actions', function() {
let actions$: Actions;
let actions$: Actions<AddAcction | SubtractAction>;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

AddAcction -> AddAction

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

@alex-okrushko
Copy link
Member

@rkirov Hey Rado. Would you be able to take a look at this PR again?

@dvitiuk-opensource
Copy link

Any update on this? I can finalize this PR, if you don't mind.

For someone who wants to try this feature in your project today, you can add this to your codebase:

declare module '@ngrx/effects' {
  export function ofType<
    V extends Extract<U, { type: T }>, 
    T extends string = string, 
    U extends Action = Action
  >(...actions: T[]): OperatorFunction<U, V>;
}

@timdeschryver
Copy link
Member

If you want I can finish this one up @rkirov but you'll have to give maintainers write access to your branch.

@rkirov
Copy link
Contributor Author

rkirov commented Oct 2, 2018

I will rebase it today. Sorry for the delay.

Btw, we have been running this with change internally in google for some time and I haven't heard any complaints.

It is based on TS2.8 introduced conditional types. Upgrade package.json
to that version.

Also remove the deprecated non-pipable version of `ofType` so that we
don't have to keep complicated types in two places.

Fixes some tests post removal of non-pipable version.

Original implementation by @mtaran-google.

BREAKING CHANGES:

Removes .ofType method on Actions. Instead use the provided 'ofType'
rxjs operator.

BEFORE:
```
this.actions.ofType('INCREMENT')
```

AFTER:

```
import { ofType } from '@ngrx/store';
...
this.action.pipe(ofType('INCREMENT'))
```
@rkirov
Copy link
Contributor Author

rkirov commented Oct 3, 2018

Rebased, removed compat bits and updated commit message with 'BREAKING CHANGE'. Please take another look.

Copy link
Member

@timdeschryver timdeschryver left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍

@brandonroberts brandonroberts merged commit 8d56a6f into ngrx:master Oct 16, 2018
@brandonroberts
Copy link
Member

Thanks @rkirov!

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

Successfully merging this pull request may close these issues.

8 participants