Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

fix(loader): module.decorator order of operations is now irrelevant #14348

Closed
wants to merge 1 commit into from

Conversation

robertmirro
Copy link
Contributor

What kind of change does this PR introduce? (Bug fix, feature, docs update, ...)
Bug fix

What is the current behavior? (You can also link to an open issue here)
See #12382

What is the new behavior (if this is a feature change)?
module.decorator declarations are now invoked via the configBlocks queue to ensure that the provider being decorated has been declared regardless of the order in which module.decorator is declared.

Does this PR introduce a breaking change?
Possibly. If the same provider is decorated multiple times via a combination of module.decorator and $provide.decorator (via module.config), the decoration order may change now depending on declaration order of the module methods.

Please check if the PR fulfills these requirements

Other information:

I didn't ngdoc the added queue param of invokeLaterAndSetModuleName because this fn isn't publically exposed and queue wasn't doc'd for invokeLater.

module.decorator() is now processed via the configBlocks order of operations and:

  • no longer throws if declared before the provider being decorated
  • guarantees the correct provider will be decorated when multiple,
    same-name providers are defined

Closes: #12382

@Narretz
Copy link
Contributor

Narretz commented Apr 7, 2016

Thanks for the PR. I do think we should add a BC note to be on the safe side. Can you please add one, that describes the new behavior, how it's different from the old behavior and how this might break existing code? Small code example (that shows how order affects execution) would also be nice. You can have a look at the Changelog or the Migration guide to see how a BC note should look.

@robertmirro
Copy link
Contributor Author

How? Amend commit 478650c and re-push? If so, won't that wipe out your comments above?

@Narretz
Copy link
Contributor

Narretz commented Apr 8, 2016

Yes amend and force push

@Narretz
Copy link
Contributor

Narretz commented Apr 8, 2016

The comments are independent of that. And comments on changes are also preserved if the code doesn't change

@robertmirro robertmirro force-pushed the rjm-issue-12382 branch 2 times, most recently from 7a1750a to e96c44c Compare April 8, 2016 21:55
@robertmirro
Copy link
Contributor Author

Thanks! Please let me know if the new commit is not what you had in mind.

@Narretz
Copy link
Contributor

Narretz commented Apr 13, 2016

The commit message looks great, very nice!

I would only change the names anotherDecoratorFn and theDecoratorFn to provideDecoratorFn and moduleDecoratorFn to make it easier to see the new order.

And please move the Closes #... above the BC notices, so it doesn't show up in it.

I'd also like to see a test that ensures defining a decorator before its provider works.

@robertmirro
Copy link
Contributor Author

I've made the following changes:

  • relocated the Closes tag to come before BREAKING CHANGE: in the commit message
  • renamed fn name examples in the commit message
  • created an additional unit test

Regarding the additional unit test...

I included the additional unit test in loaderSpec (as opposed to injectorSpec) because the failure described via #12382 is relevant to module.

However, I'm not entirely convinced this test is necessary because this PR is meant to move the processing of module.decorator declarations to the configBlocks queue. The originally modified unit test in loaderSpec proves that change works.

Regarding the Closes tag...

Neither the Contributing to AngularJS nor the AngularJS Git Commit Message Conventions documents seem to specify that the Closes #<issue number> tag needs to come before BREAKING CHANGE:. In fact, the outline section displayed at the top of AngularJS Git Commit Message Conventions seems to indicate Referencing issues comes last in the Message footer.

Many BC commits seem to follow this pattern: d06431e, 98c2db7, 98528be, 983b059, etc...

Maybe the docs can be updated if format is critical and the above is no longer acceptable?

Thanks!

@Narretz
Copy link
Contributor

Narretz commented May 6, 2016

There's a typo in the new loaderSpec test that makes the tests fail.
I know that the test you changed validates the correctness of the fix, but it's important (to me) to have a test that exactly tests this change. The changed test doesn't make it obvious that the order doesn't matter now.

Wrt to the commit message guidelines, putting Closes #... before the BC notice makes the CL output prettier. But we could add that to the guidelines.

`module.decorator` is now processed via the configBlocks order of operations and:
  1. no longer throws error if declared before the provider being decorated.
  2. guarantees the correct provider will be decorated when multiple, same-name
  providers are defined.

(1) Prior to this fix, declaring `module.decorator` before the provider that it
decorates results in throwing an error:

```js
angular
  .module('theApp', [])
  .decorator('theFactory', moduleDecoratorFn)
  .factory('theFactory', theFactoryFn);

// Error: [$injector:modulerr] Failed to instantiate module theApp due to:
// Error: [$injector:unpr] Unknown provider: theFactoryProvider
```

The result of this fix now allows for the declaration order above.

(2) Prior to this fix, declaring `module.decorator` before the final, same-named
provider results in that provider **not** being decorated as expected:

**NOTE:** Angular does not use provider name spacing, so the final declared
provider is selected if multiple, same-named providers are declared.

```js
angular
  .module('theApp', [])
  .factory('theFactory', theFactoryFn)
  .decorator('theFactory', moduleDecoratorFn)
  .factory('theFactory', theOtherFactoryFn);

// `theOtherFactoryFn` is selected as 'theFactory' provider but it is **not**
// decorated via `moduleDecoratorFn` as expected.
```

The result of this fix ensures that `theOtherFactoryFn` will be decorated as
expected when using the declaration order above.

Closes angular#12382

BREAKING CHANGE:

`module.decorator` declarations are now processed as part of the `module.config`
queue and may result in providers being decorated in a different order if
`module.config` blocks are also used to decorate providers via
`$provide.decorator`.

For example, consider the following declaration order in which 'theFactory' is
decorated by both a `module.decorator` and a `$provide.decorator`:

```js
angular
  .module('theApp', [])
  .factory('theFactory', theFactoryFn)
  .config(function($provide) {
    $provide.decorator('theFactory', provideDecoratorFn);
  })
  .decorator('theFactory', moduleDecoratorFn);
```

Prior to this fix, 'theFactory' provider would be decorated in the following
order:
  1. moduleDecoratorFn
  2. provideDecoratorFn

The result of this fix changes the order in which 'theFactory' is decorated
because now `module.decorator` declarations are processed in the same order as
`module.config` declarations:
  1. provideDecoratorFn
  2. moduleDecoratorFn
@robertmirro
Copy link
Contributor Author

Oops, regarding the typo...

I had been testing via grunt test:unit all along.

I changed naming conventions that are used in the test at the last moment and then ran grunt test as a final sanity check before my commit.

Unfortunately the package task that is invoked before test.unit stalls on minall (more specifically, I had to remove the parseext item from min in order to resolve the stall in the test process), so I didn't noticed that test.unit was never re-run.

This behavior occurs on Win7 Pro 64-bit.

@gkalpak
Copy link
Member

gkalpak commented May 11, 2016

@robertmirro: Are you sure it "stalls". It usually takes a lot of time and CPU (and can give the impression that it's frozen), but it's not.

@robertmirro
Copy link
Contributor Author

@gkalpak I just waited 5-ish minutes. That feels "stalled" to me, but I'm open to using a different word to describe this behavior. 😄

With parseext included, minall never completes after angular.min.js is generated:
http://screencast.com/t/dhEVZizmxc25

With parseext commented out, minall completes and the next task (collect-errors) begins:
http://screencast.com/t/tD7StOaSML

@gkalpak
Copy link
Member

gkalpak commented May 13, 2016

That's odd (unless it just needs more time on your machine - depending on the available resources).
The tasks are running fine on my Windows 10 😃

If I were you I would (1) try giving it some more time and (2) removing node_modules and re-installing (I guess that's the "switch it off and back on" strategy in the Node.js world 😛)

@robertmirro
Copy link
Contributor Author

Being that I'm no stranger to anomalies such as this, I'll choose option (3)... Leaving parseext commented out while simultaneously attempting to forget that this issue probably only exists on my machine. 😜

Thanks though!

@gkalpak
Copy link
Member

gkalpak commented May 14, 2016

Yeah that would work as well, I guess 😃

@Narretz Narretz closed this in 6a2ebdb May 24, 2016
sjbarker added a commit to sjbarker/angular.js that referenced this pull request May 31, 2016
Commit 6a2ebdb removes the caveat of declaring a provider prior to
decorating it through the module.decorator function.

Remove statements warning of the prior requirement for order of
operations.

See angular#12382, PR angular#14348, and [angular#14372 comment 206452412]
(angular#14372 (comment))
sjbarker added a commit to sjbarker/angular.js that referenced this pull request May 31, 2016
Commit 6a2ebdb removes the caveat of declaring a provider prior to
decorating it through the module.decorator function.

Remove statements warning of the prior requirement for order of
operations.

See angular#12382, PR angular#14348, and [angular#14372 comment 206452412]
(angular#14372 (comment))
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

module.decorator() requires explicit order of operations.
4 participants