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

feat(http): Add 'useLegacyMethods' to $httpProvider #12112

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 10 additions & 0 deletions docs/content/error/$http/noerror.ngdoc
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
@ngdoc error
@name $http:noerror
@fullName The method `error` on the $http result has been disabled.
@description

This error occurs when the legacy promise extensions {@link ng.$http `$http`} have been disabled.

To resolve this error, either change to code to use `then` or add `$httpProvider.useLegacyPromiseExtensions(true);` to your application.

For more information, see the {@link ng.$http `$http`} service API documentation.
10 changes: 10 additions & 0 deletions docs/content/error/$http/nosuccess.ngdoc
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
@ngdoc error
@name $http:nosuccess
@fullName The method `success` on the $http result has been disabled.
@description

This error occurs when the legacy promise extensions {@link ng.$http `$http`} have been disabled.

To resolve this error, either change to code to use `then` or add `$httpProvider.useLegacyPromiseExtensions(true);` to your application.

For more information, see the {@link ng.$http `$http`} service API documentation.
117 changes: 74 additions & 43 deletions src/ng/http.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ var JSON_ENDS = {
'{': /}$/
};
var JSON_PROTECTION_PREFIX = /^\)\]\}',?\n/;
var $httpMinErr = minErr('$http');

function serializeValue(v) {
if (isObject(v)) {
Expand Down Expand Up @@ -330,6 +331,29 @@ function $HttpProvider() {
return useApplyAsync;
};

var useHttpPromise = true;
/**
* @ngdoc method
* @name $httpProvider#useLegacyPromiseExtensions
*
* Configure $http service to return promises without the shorthand methods `success` and `error`. It should
* be used to make sure that applications work without these methods.
*
* Defaults to false. If no value is specified, returns the current configured value.
*
* @param {boolean=} value If true, $http will return a promise without the `success` and `error methods.
*
* @returns {boolean|Object} If a value is specified, returns the $httpProvider for chaining.
* otherwise, returns the current configured value.
**/
this.useLegacyPromiseExtensions = function(value) {
if (isDefined(value)) {
useHttpPromise = !!value;
return this;
}
return useHttpPromise;
};

/**
* @ngdoc property
* @name $httpProvider#interceptors
Expand Down Expand Up @@ -396,17 +420,15 @@ function $HttpProvider() {
*
* ## General usage
* The `$http` service is a function which takes a single argument — a configuration object —
* that is used to generate an HTTP request and returns a {@link ng.$q promise}
* with two $http specific methods: `success` and `error`.
* that is used to generate an HTTP request and returns a {@link ng.$q promise}.
*
* ```js
* // Simple GET request example :
* $http.get('/someUrl').
* success(function(data, status, headers, config) {
* then(function(response) {
* // this callback will be called asynchronously
* // when the response is available
* }).
* error(function(data, status, headers, config) {
* }, function(response) {
* // called asynchronously if an error occurs
* // or server returns response with an error status.
* });
Expand All @@ -415,21 +437,23 @@ function $HttpProvider() {
* ```js
* // Simple POST request example (passing data) :
* $http.post('/someUrl', {msg:'hello word!'}).
* success(function(data, status, headers, config) {
* then(function(response) {
* // this callback will be called asynchronously
* // when the response is available
* }).
* error(function(data, status, headers, config) {
* }, function(response) {
* // called asynchronously if an error occurs
* // or server returns response with an error status.
* });
* ```
*
* The response object has these properties:
*
* Since the returned value of calling the $http function is a `promise`, you can also use
* the `then` method to register callbacks, and these callbacks will receive a single argument –
* an object representing the response. See the API signature and type info below for more
* details.
* - **data** – `{string|Object}` – The response body transformed with the transform
* functions.
* - **status** – `{number}` – HTTP status code of the response.
* - **headers** – `{function([headerName])}` – Header getter function.
* - **config** – `{Object}` – The configuration object that was used to generate the request.
* - **statusText** – `{string}` – HTTP status text of the response.
*
* A response status code between 200 and 299 is considered a success status and
* will result in the success callback being called. Note that if the response is a redirect,
Expand All @@ -453,8 +477,8 @@ function $HttpProvider() {
* request data must be passed in for POST/PUT requests.
*
* ```js
* $http.get('/someUrl').success(successCallback);
* $http.post('/someUrl', data).success(successCallback);
* $http.get('/someUrl').then(successCallback);
* $http.post('/someUrl', data).then(successCallback);
* ```
*
* Complete list of shortcut methods:
Expand Down Expand Up @@ -511,7 +535,7 @@ function $HttpProvider() {
* data: { test: 'test' }
* }
*
* $http(req).success(function(){...}).error(function(){...});
* $http(req).then(function(){...}, function(){...});
* ```
*
* ## Transforming Requests and Responses
Expand Down Expand Up @@ -794,14 +818,12 @@ function $HttpProvider() {
* response object. The `success` and `error` methods take a single argument - a function that
* will be called when the request succeeds or fails respectively. The arguments passed into
* these functions are destructured representation of the response object passed into the
* `then` method. The response object has these properties:
* `then` method.
*
* - **data** – `{string|Object}` – The response body transformed with the transform
* functions.
* - **status** – `{number}` – HTTP status code of the response.
* - **headers** – `{function([headerName])}` – Header getter function.
* - **config** – `{Object}` – The configuration object that was used to generate the request.
* - **statusText** – `{string}` – HTTP status text of the response.
* <div class="alert alert-error">
* **Note:** the short hand methods `success` and `error` are deprecated.
* Use the standard `then` method instead.
* </div>
*
* @property {Array.<Object>} pendingRequests Array of config objects for currently pending
* requests. This is primarily meant to be used for debugging purposes.
Expand Down Expand Up @@ -843,13 +865,12 @@ function $HttpProvider() {
$scope.response = null;

$http({method: $scope.method, url: $scope.url, cache: $templateCache}).
success(function(data, status) {
$scope.status = status;
$scope.data = data;
}).
error(function(data, status) {
$scope.data = data || "Request failed";
$scope.status = status;
then(function(response) {
$scope.status = response.status;
$scope.data = response.data;
}, function(response) {
$scope.data = response.data || "Request failed";
$scope.status = response.status;
});
};

Expand Down Expand Up @@ -954,23 +975,33 @@ function $HttpProvider() {
promise = promise.then(thenFn, rejectFn);
}

promise.success = function(fn) {
assertArgFn(fn, 'fn');
if (useHttpPromise) {
promise.success = function(fn) {
assertArgFn(fn, 'fn');

promise.then(function(response) {
fn(response.data, response.status, response.headers, config);
});
return promise;
};
promise.then(function(response) {
fn(response.data, response.status, response.headers, config);
});
return promise;
};

promise.error = function(fn) {
assertArgFn(fn, 'fn');
promise.error = function(fn) {
assertArgFn(fn, 'fn');

promise.then(null, function(response) {
fn(response.data, response.status, response.headers, config);
});
return promise;
};
promise.then(null, function(response) {
fn(response.data, response.status, response.headers, config);
});
return promise;
};
} else {
promise.success = function() {
throw $httpMinErr('nosuccess', 'The method `success` on the $http result has been disabled.');
};

promise.error = function() {
throw $httpMinErr('noerror', 'The method `error` on the $http result has been disabled.');
};
}

return promise;

Expand Down
30 changes: 30 additions & 0 deletions test/ng/httpSpec.js
Original file line number Diff line number Diff line change
Expand Up @@ -1975,6 +1975,36 @@ describe('$http with $applyAsync', function() {
});
});

describe('$http without useLegacyPromiseExtensions', function() {
var $httpBackend, $http;
beforeEach(module(function($httpProvider) {
$httpProvider.useLegacyPromiseExtensions(false);
}, provideLog));

beforeEach(inject(['$httpBackend', '$http', '$rootScope', function($hb, $h, $rs) {
$httpBackend = $hb;
$http = $h;
}]));

it('should throw when the success or error methods are called if useLegacyPromiseExtensions is false', function() {
$httpBackend.expect('GET', '/url').respond('');
var promise = $http({url: '/url'});

function callSucess() {
promise.success();
}

function callError() {
promise.error();
}

expect(callSucess).toThrowMinErr(
'$http', 'nosuccess', 'The method `success` on the $http result has been disabled.');
expect(callError).toThrowMinErr(
'$http', 'noerror', 'The method `error` on the $http result has been disabled.');
});
});

describe('$http param serializers', function() {
Copy link
Contributor

Choose a reason for hiding this comment

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

We also need a test of throwing the error if the legacy methods are disabled.


var defSer, jqrSer;
Expand Down