diff --git a/docs/content/error/$http/noerror.ngdoc b/docs/content/error/$http/noerror.ngdoc new file mode 100644 index 000000000000..00105c13507e --- /dev/null +++ b/docs/content/error/$http/noerror.ngdoc @@ -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. diff --git a/docs/content/error/$http/nosuccess.ngdoc b/docs/content/error/$http/nosuccess.ngdoc new file mode 100644 index 000000000000..35d8574337f0 --- /dev/null +++ b/docs/content/error/$http/nosuccess.ngdoc @@ -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. diff --git a/src/ng/http.js b/src/ng/http.js index 9070c47bfd90..9b4f0a2e6bc4 100644 --- a/src/ng/http.js +++ b/src/ng/http.js @@ -8,6 +8,7 @@ var JSON_ENDS = { '{': /}$/ }; var JSON_PROTECTION_PREFIX = /^\)\]\}',?\n/; +var $httpMinErr = minErr('$http'); function serializeValue(v) { if (isObject(v)) { @@ -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 @@ -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. * }); @@ -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, @@ -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: @@ -511,7 +535,7 @@ function $HttpProvider() { * data: { test: 'test' } * } * - * $http(req).success(function(){...}).error(function(){...}); + * $http(req).then(function(){...}, function(){...}); * ``` * * ## Transforming Requests and Responses @@ -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. + *
+ * **Note:** the short hand methods `success` and `error` are deprecated. + * Use the standard `then` method instead. + *
* * @property {Array.} pendingRequests Array of config objects for currently pending * requests. This is primarily meant to be used for debugging purposes. @@ -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; }); }; @@ -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; diff --git a/test/ng/httpSpec.js b/test/ng/httpSpec.js index ab4a9f43e878..27dcc93767a8 100644 --- a/test/ng/httpSpec.js +++ b/test/ng/httpSpec.js @@ -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() { var defSer, jqrSer;