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

fix(resource): check whether response matches action.isArray #3054

Closed

Conversation

petebacondarwin
Copy link
Contributor

When using $resource you must setup your actions carefully based on what the server returns. If the server responds to a request with an array then you must configure the action with isArray:true and vice versa. The built-in get action defaults to isArray:false and the query action defaults to isArray:true, which is must be changed if the server does not do this.
Before this change, the error message was an exception inside angular.copy, which didn't explain what the real problem was.
Rather than changing the way that angular.copy works, this change ensures that a better error message is provided to the programmer if they do not set up their resource actions correctly.

Closes #2255, #1044

When using $resource you must setup your actions carefully based on what the server returns. If the server responds to a request with an array then you must configure the action with `isArray:true` and vice versa.  The built-in `get` action defaults to `isArray:false` and the `query` action defaults to `isArray:true`, which is must be changed if the server does not do this.
Before the error message was an exception inside angular.copy, which didn't explain what the real problem was.
Rather than changing the way that angular.copy works, this change ensures that a better error message is provided to the programmer if they do not set up their resource actions correctly.

Closes angular#2255, angular#1044
@btford
Copy link
Contributor

btford commented Jul 12, 2013

LGTM. Would like to run this by @IgorMinar before merging.

@@ -472,6 +472,10 @@ angular.module('ngResource', ['ng']).
promise = value.$promise;

if (data) {
if ( angular.isArray(data) != !!action.isArray ) {
return $q.reject("Error in resource configuration. Expected response to contain an " +
(action.isArray?'array':'object') + " but got an " + (angular.isArray(data)?'array':'object'));
Copy link
Contributor

Choose a reason for hiding this comment

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

can we throw a minErr instead of explicitly rejecting the promise?

@IgorMinar
Copy link
Contributor

Nice fix, but please consider throwing minErr instead

@petebacondarwin
Copy link
Contributor Author

Will throwing the minErr be converted to a $q rejection?

On 14 July 2013 05:52, Igor Minar notifications@github.com wrote:

Nice fix, but please consider throwing minErr instead


Reply to this email directly or view it on GitHubhttps://github.com//pull/3054#issuecomment-20931696
.

@ghost
Copy link

ghost commented Jul 18, 2013

bump!

@IgorMinar
Copy link
Contributor

@petebacondarwin I spoke with @ksheedlo and asked him to merge this one as well as #3159 in after some modifications

@petebacondarwin
Copy link
Contributor Author

Great!

@ksheedlo
Copy link
Contributor

I merged #3159 yesterday, but this one is giving me some trouble. I checked it out and changed the return $q.reject to a throw ngResourceMinErr and I'm getting strange test failures:

Chrome 28.0.1500 (Linux) resource should fail if action expects an object but response is an array FAILED
    Error: [ngResource:nocfg] Error in resource configuration. Expected response to contain an array but got an object
        at /usr/local/google/home/sheedlo/angular/angular.js/src/minErr.js:55:12
        at $http.then.value.$resolved (/usr/local/google/home/sheedlo/angular/angular.js/src/ngResource/resource.js:477:23)
        at wrappedCallback (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/q.js:238:59)
        at wrappedCallback (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/q.js:238:59)
        at /usr/local/google/home/sheedlo/angular/angular.js/src/ng/q.js:319:26
        at Object.Scope.$eval (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:653:28)
        at Object.Scope.$digest (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:509:23)
        at Object.Scope.$apply (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:739:24)
        at done (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/http.js:976:45)
        at handleResponse (/usr/local/google/home/sheedlo/angular/angular.js/src/ngMock/angular-mocks.js:985:9)
    Error: [$rootScope:inprog] $digest already in progress
        at /usr/local/google/home/sheedlo/angular/angular.js/src/minErr.js:55:12
        at beginPhase (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:930:15)
        at Object.Scope.$digest (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:501:9)
        at Function.$httpBackend.verifyNoOutstandingExpectation (/usr/local/google/home/sheedlo/angular/angular.js/src/ngMock/angular-mocks.js:1325:16)
        at null.<anonymous> (/usr/local/google/home/sheedlo/angular/angular.js/test/ngResource/resourceSpec.js:31:18)
Chrome 28.0.1500 (Linux) resource should fail if action expects an array but response is an object FAILED
    Error: [ngResource:nocfg] Error in resource configuration. Expected response to contain an object but got an array
        at /usr/local/google/home/sheedlo/angular/angular.js/src/minErr.js:55:12
        at $http.then.value.$resolved (/usr/local/google/home/sheedlo/angular/angular.js/src/ngResource/resource.js:477:23)
        at wrappedCallback (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/q.js:238:59)
        at wrappedCallback (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/q.js:238:59)
        at /usr/local/google/home/sheedlo/angular/angular.js/src/ng/q.js:319:26
        at Object.Scope.$eval (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:653:28)
        at Object.Scope.$digest (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:509:23)
        at Object.Scope.$apply (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:739:24)
        at done (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/http.js:976:45)
        at handleResponse (/usr/local/google/home/sheedlo/angular/angular.js/src/ngMock/angular-mocks.js:985:9)
    Error: [$rootScope:inprog] $digest already in progress
        at /usr/local/google/home/sheedlo/angular/angular.js/src/minErr.js:55:12
        at beginPhase (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:930:15)
        at Object.Scope.$digest (/usr/local/google/home/sheedlo/angular/angular.js/src/ng/rootScope.js:501:9)
        at Function.$httpBackend.verifyNoOutstandingExpectation (/usr/local/google/home/sheedlo/angular/angular.js/src/ngMock/angular-mocks.js:1325:16)
        at null.<anonymous> (/usr/local/google/home/sheedlo/angular/angular.js/test/ngResource/resourceSpec.js:31:18)

An error thrown inside the promise should be converted to a $q.reject internally. The stack trace shows that it's not happening. The other odd thing about this is that the failure is occurring in afterEach when we call $httpBackend.verifyNoOutstandingExpectation. So the expectations are passing in the body of the tests.

I don't have time to go in and debug this right now. There's a chance I could get to it later this afternoon or sometime next week, but if someone else wants to take a look, that would be appreciated.

@ksheedlo
Copy link
Contributor

If anyone else is going to try and reproduce the behavior I'm getting, I've published a branch: https://github.com/ksheedlo/angular.js/tree/pr3054

@petebacondarwin
Copy link
Contributor Author

Is it that you are not yet in a promise handler?

Pete
...from my mobile.
On 19 Jul 2013 21:29, "Ken Sheedlo" notifications@github.com wrote:

If anyone else is going to try and reproduce the behavior I'm getting,
I've published a branch:
https://github.com/ksheedlo/angular.js/tree/pr3054


Reply to this email directly or view it on GitHubhttps://github.com//pull/3054#issuecomment-21275365
.

@petebacondarwin
Copy link
Contributor Author

Hey @ksheedlo - I think you have uncovered a pretty major bug in $q!
On this line: https://github.com/angular/angular.js/blob/master/src/ng/q.js#L240 and this line: https://github.com/angular/angular.js/blob/master/src/ng/q.js#L249, exceptionHandler(e) is called before result.reject(e), which actually re-throws the error and so reject() is never called.

This is not picked up in the $q spec since $exceptionHandler is mocked out with effectively noop()

@petebacondarwin
Copy link
Contributor Author

Actually the problem is simply that, by default, ngMock.$exceptionHandlerProvider is set to rethrow unless told otherwise (i.e. to "log"). Many specs do this: notably ngRepeat and $http. I propose that we do this here too.

@petebacondarwin
Copy link
Contributor Author

Closing in favour of #3400

@petebacondarwin petebacondarwin deleted the resource-err-msg branch November 7, 2013 11:57
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants