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

improve interpolation of params in ngRoute #8200

Closed
wants to merge 2 commits into from

Conversation

avdv
Copy link

@avdv avdv commented Jul 15, 2014

Request Type: bug

How to reproduce: 6f77610

  1. create a route with optional parameters, eg. $routeProvider.when('/profile/:userId?', ...)
  2. angular will add the automatic redirect with or without the trailing /, in this case '/profile/userId?/'
  3. point your browser to the redirect http://localhost/#!/profile/0/, you'll see the encoded ? (%3F) at the end of the url: http://localhost/#!/profile/0%3F

Component(s): ngRoute

Impact: medium

Complexity: small

This issue is related to:

Detailed Description:

In line with #5746, this changes how parameters are interpolated in redirectTo strings.

First, I added a test case (it's my first in Javascript ever, so bear with me) which exhibits the failure:

        Expected '/bar/id1/subid3?/23' to equal '/bar/id1/subid3/23'.
        Error: Expected '/bar/id1/subid3?/23' to equal '/bar/id1/subid3/23'.
            at null.<anonymous> (/home/vtc/src/angular.js/test/ngRoute/routeSpec.js:826:34)
            at Object.invoke (/home/vtc/src/angular.js/src/auto/injector.js:801:17)
            at workFn (/home/vtc/src/angular.js/src/ngMock/angular-mocks.js:2187:20)
            at window.inject.angular.mock.inject (/home/vtc/src/angular.js/src/ngMock/angular-mocks.js:2159:42)
            at null.<anonymous> (/home/vtc/src/angular.js/test/ngRoute/routeSpec.js:822:7)
        Expected '/bar/id1/?/23' to equal '/bar/id1/23'.
        Error: Expected '/bar/id1/?/23' to equal '/bar/id1/23'.
            at null.<anonymous> (/home/vtc/src/angular.js/test/ngRoute/routeSpec.js:832:34)
            at Object.invoke (/home/vtc/src/angular.js/src/auto/injector.js:801:17)
            at workFn (/home/vtc/src/angular.js/src/ngMock/angular-mocks.js:2187:20)
            at window.inject.angular.mock.inject (/home/vtc/src/angular.js/src/ngMock/angular-mocks.js:2159:42)
            at null.<anonymous> (/home/vtc/src/angular.js/test/ngRoute/routeSpec.js:822:7)
        Expected '/path/foovalue/barvalue*/1' to equal '/path/foovalue/barvalue/1'.
        Error: Expected '/path/foovalue/barvalue*/1' to equal '/path/foovalue/barvalue/1'.
            at null.<anonymous> (/home/vtc/src/angular.js/test/ngRoute/routeSpec.js:837:34)
            at Object.invoke (/home/vtc/src/angular.js/src/auto/injector.js:801:17)
            at workFn (/home/vtc/src/angular.js/src/ngMock/angular-mocks.js:2187:20)
            at window.inject.angular.mock.inject (/home/vtc/src/angular.js/src/ngMock/angular-mocks.js:2159:42)
            at null.<anonymous> (/home/vtc/src/angular.js/test/ngRoute/routeSpec.js:822:7)

Alas, the fix from Pasvaz in #5746 lead to an error too:

        Expected '/bar/id1//23' to equal '/bar/id1/23'.
        Error: Expected '/bar/id1//23' to equal '/bar/id1/23'.
            at null.<anonymous> (/home/vtc/src/angular.js/test/ngRoute/routeSpec.js:832:34)
            at Object.invoke (/home/vtc/src/angular.js/src/auto/injector.js:801:17)
            at workFn (/home/vtc/src/angular.js/src/ngMock/angular-mocks.js:2187:20)
            at window.inject.angular.mock.inject (/home/vtc/src/angular.js/src/ngMock/angular-mocks.js:2159:42)
            at null.<anonymous> (/home/vtc/src/angular.js/test/ngRoute/routeSpec.js:822:7)l

Note the doubled slash, which is kind of awkward. So, a slash should be removed when the optional parameter is missing.

Now string interpolation is done similar to the generation of the regexp in route, using a replace function.

All good, tests pass.

Other Comments:

I don't know what your preference is, should I reorder the commits in order to keep the tests passing for every single commit?

@mary-poppins
Copy link

Thanks for the PR! Please check the items below to help us merge this faster. See the contributing docs for more information.

  • Uses the issue template (#8200)

If you need to make changes to your pull request, you can update the commit with git commit --amend.
Then, update the pull request with git push -f.

Thanks again for your help!

@Narretz
Copy link
Contributor

Narretz commented Jul 15, 2014

How to reproduce points to a commit, is this intentional?

@avdv
Copy link
Author

avdv commented Jul 15, 2014

How to reproduce points to a commit, is this intentional?

I'm not sure what you're asking here. Are you talking about the commit reference I gave? What did you expect in place of it?

Basically, you just check out that commit as usual git checkout 6f77610 if you want to reproduce it.

@caitp
Copy link
Contributor

caitp commented Jul 15, 2014

@avdv I think what he's saying is, please provide some actual steps to reproduce this (IE, code to run). Posting a revision to test is helpful, but we need more than that.

edit: I see there are instructions, but a minimal code example on plnkr or heroku would be better

@Narretz
Copy link
Contributor

Narretz commented Jul 15, 2014

Oh, that's what you mean @avdv .As a reproduction I understand a runnable example, such as in a plnkr.co or maybe even a git repo to check out.

@avdv
Copy link
Author

avdv commented Jul 15, 2014

Got it. If you want that I could probably do it tomorrow...

@avdv
Copy link
Author

avdv commented Jul 16, 2014

Here you go: http://plnkr.co/edit/gO7kz2UICfErehcFPhOu?p=preview

  1. click on the "Test 1" link
  2. look at the page route parameter, it's value ends with an ?
  3. click on the "Test 2" link
  4. look at the chapterId parameter, it's value ends with an *

@loddit
Copy link

loddit commented Jul 29, 2014

I have this bug too.
how to reproduce?

  1. create a route with optional parameters, e.g. when(/:options?)
  2. go to a matched url ,then angular works well.
  3. refresh this page, you can see url change at address bar. first add a tail slash , second slash become to "%3f" ,and refresh again , add one more '%3f' => /path%3f%3f

@caitp
Copy link
Contributor

caitp commented Aug 1, 2014

Interesting --- this might be a regression, because this is documented as a feature which should work. Surprised tests aren't broken because of it. (I haven't looked too closely at the repro yet, it could be that something is just wrong there) but so far this doesn't look unreasonable.

@@ -813,6 +813,32 @@ describe('$route', function() {
});


it('should interpolate optional and long route vars in the redirected path from original path', function() {
Copy link
Contributor

Choose a reason for hiding this comment

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

what the heck does "long route vars" mean here?

Copy link
Author

Choose a reason for hiding this comment

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

what the heck does "long route vars" mean here?

I mean those vars with an asterisk at the end. How do you call them? Greedy route vars?

May be it's a bit confusing that the documentation calls those things "named groups", but throughout the code the term "route params" as well as "route vars" is used?

Copy link
Contributor

Choose a reason for hiding this comment

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

The test case for that feature calls it a "catch-all" parameter, but the documentation refers to it as "eagerly matches all characters" (whereas without the '*', we match until the next slash).

@caitp
Copy link
Contributor

caitp commented Aug 1, 2014

So, if you could address those comments, that would be helpful -- if someone else says this looks good (with comments addressed) I think we can merge it

@avdv
Copy link
Author

avdv commented Aug 2, 2014

I have this bug too.
how to reproduce?

create a route with optional parameters, e.g. when(/:options?)
go to a matched url ,then angular works well.
refresh this page, you can see url change at address bar. first add a tail slash , second slash become to "%3f" ,and refresh again , add one more '%3f' => /path%3f%3f

Have you tried this patch and did it fix it this problem for you?

@loddit
Copy link

loddit commented Aug 4, 2014

@avdv yes, it`s fix this problem.
I pull your branch, grunt buld , and copy angular-route.js to bower_components folder of my app.

I`m not sure about this pull request. the new spec looks like about path resolution.

I just fix this bug ,to avoid route tread the "?" after ":option" as a "/.*/" https://gist.github.com/loddit/2e97ea12a32f12d7427f

$rootScope.$digest();

expect($location.path()).toEqual('/bar/id1/23');
expect($location.search()).toEqual({extraId: 'gah'});
Copy link
Contributor

Choose a reason for hiding this comment

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

should test that $route.current.params equal what we expect, not just $location.search() (for each of these cases).

@caitp
Copy link
Contributor

caitp commented Aug 4, 2014

I do expect this change to be a little bit slower than the current implementation, so it would be good to consider alternative ways of implementing it. But the test case is good stuff, just needs to be broken into smaller pieces.

@avdv
Copy link
Author

avdv commented Aug 4, 2014

@caitp Thanks for your review. I think I (hopefully) addressed all of your complains.

\edit: seems the Travis build is a bit flaky at times. Is there some way to trigger a re-run?

avdv added 2 commits August 20, 2014 17:11
Make sure that :name? and :name* vars are properly interpolated in redirectUrl.
When the redirect path contains optional or special parameters the
interpolation function misses to remove them.

This is also the case for the auto-generated redirection where a slash
is appended or removed to a route.

Change the way parameters are interpolated in a string. Remove undefined
optional named groups and properly prepend a slash in case the named
group was starting with one.

Related to angular#5746.
@avdv avdv force-pushed the ngRoute-improve-interpolation branch from 9fcdb38 to 0336d57 Compare August 20, 2014 15:14
@avdv
Copy link
Author

avdv commented Aug 20, 2014

I did just rebase to upstream master which triggered a new Travis CI build.

Any comments on the current patch set?

@vladimir-vg
Copy link

This bug also causes problems when using recently added updateParams to $route: #8358

Getting %3F for /foo/:bar? like routes.

Will it be merged to 1.3.0 ?

vladimir-vg added a commit to clickavia/bower-angular-route that referenced this pull request Oct 6, 2014
Problem already fixed in angular/angular.js#8200 ,
but still doesn't merged to upstream. This commit fixes bug in bower package
for deployment-only. Hope fix mentioned above will be merged soon, and this
repo will be purged.
vladimir-vg added a commit to clickavia/bower-angular-route that referenced this pull request Oct 6, 2014
Problem already fixed in angular/angular.js#8200 ,
but still doesn't merged to upstream. This commit fixes bug in bower package
for deployment-only. Hope fix mentioned above will be merged soon, and this
repo will be purged.
@jeffbcross jeffbcross force-pushed the master branch 2 times, most recently from abdaab7 to 30996f8 Compare October 8, 2014 19:46
@jeffbcross jeffbcross force-pushed the master branch 2 times, most recently from e8dc429 to e83fab9 Compare October 10, 2014 17:37
@Narretz Narretz modified the milestones: 1.3.x, Backlog Oct 22, 2014
@Narretz
Copy link
Contributor

Narretz commented Oct 22, 2014

Easier approach that's probably not working (#5746 (comment)) as-is here #5746

@Narretz
Copy link
Contributor

Narretz commented Feb 8, 2015

This was fixed by 891acf4 back in November

@Narretz Narretz closed this Feb 8, 2015
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.

7 participants