parent
e68e69e7e5
commit
5b5ffe75d0
|
@ -10,8 +10,9 @@ module.exports = new Package('angular-public', [basePackage])
|
||||||
'angular2/core.ts',
|
'angular2/core.ts',
|
||||||
'angular2/di.ts',
|
'angular2/di.ts',
|
||||||
'angular2/directives.ts',
|
'angular2/directives.ts',
|
||||||
|
'angular2/http.ts',
|
||||||
'angular2/forms.ts',
|
'angular2/forms.ts',
|
||||||
'angular2/router.js',
|
'angular2/router.ts',
|
||||||
'angular2/test.ts',
|
'angular2/test.ts',
|
||||||
'angular2/pipes.ts'
|
'angular2/pipes.ts'
|
||||||
];
|
];
|
||||||
|
|
|
@ -4,6 +4,7 @@ export * from './annotations';
|
||||||
export * from './directives';
|
export * from './directives';
|
||||||
export * from './forms';
|
export * from './forms';
|
||||||
export * from './di';
|
export * from './di';
|
||||||
|
export * from './http';
|
||||||
export {Observable, EventEmitter} from 'angular2/src/facade/async';
|
export {Observable, EventEmitter} from 'angular2/src/facade/async';
|
||||||
export * from 'angular2/src/render/api';
|
export * from 'angular2/src/render/api';
|
||||||
export {DomRenderer, DOCUMENT_TOKEN} from 'angular2/src/render/dom/dom_renderer';
|
export {DomRenderer, DOCUMENT_TOKEN} from 'angular2/src/render/dom/dom_renderer';
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
library angular2.http;
|
|
@ -1,10 +1,41 @@
|
||||||
|
/**
|
||||||
|
* @module
|
||||||
|
* @public
|
||||||
|
* @description
|
||||||
|
* The http module provides services to perform http requests. To get started, see the {@link Http}
|
||||||
|
* class.
|
||||||
|
*/
|
||||||
import {bind, Binding} from 'angular2/di';
|
import {bind, Binding} from 'angular2/di';
|
||||||
import {Http, HttpFactory} from './src/http/http';
|
import {Http, HttpFactory} from './src/http/http';
|
||||||
import {XHRBackend} from 'angular2/src/http/backends/xhr_backend';
|
import {XHRBackend, XHRConnection} from 'angular2/src/http/backends/xhr_backend';
|
||||||
import {BrowserXHR} from 'angular2/src/http/backends/browser_xhr';
|
import {BrowserXHR} from 'angular2/src/http/backends/browser_xhr';
|
||||||
import {BaseRequestOptions} from 'angular2/src/http/base_request_options';
|
import {BaseRequestOptions, RequestOptions} from 'angular2/src/http/base_request_options';
|
||||||
|
|
||||||
export {Http};
|
export {MockConnection, MockBackend} from 'angular2/src/http/backends/mock_backend';
|
||||||
|
export {Request} from 'angular2/src/http/static_request';
|
||||||
|
export {Response} from 'angular2/src/http/static_response';
|
||||||
|
|
||||||
|
export {Http, XHRBackend, XHRConnection, BaseRequestOptions, RequestOptions, HttpFactory};
|
||||||
|
export {IHttp} from 'angular2/src/http/interfaces';
|
||||||
|
export {Headers} from 'angular2/src/http/headers';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a basic set of injectables to use the {@link Http} service in any application.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* import {httpInjectables, Http} from 'angular2/http';
|
||||||
|
* @Component({selector: 'http-app', appInjector: [httpInjectables]})
|
||||||
|
* @View({template: '{{data}}'})
|
||||||
|
* class MyApp {
|
||||||
|
* constructor(http:Http) {
|
||||||
|
* http.request('data.txt').subscribe(res => this.data = res.text());
|
||||||
|
* }
|
||||||
|
* }
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
*/
|
||||||
export var httpInjectables: List<any> = [
|
export var httpInjectables: List<any> = [
|
||||||
bind(BrowserXHR)
|
bind(BrowserXHR)
|
||||||
.toValue(BrowserXHR),
|
.toValue(BrowserXHR),
|
||||||
|
|
|
@ -2,31 +2,40 @@ import {Injectable} from 'angular2/di';
|
||||||
import {Request} from 'angular2/src/http/static_request';
|
import {Request} from 'angular2/src/http/static_request';
|
||||||
import {Response} from 'angular2/src/http/static_response';
|
import {Response} from 'angular2/src/http/static_response';
|
||||||
import {ReadyStates} from 'angular2/src/http/enums';
|
import {ReadyStates} from 'angular2/src/http/enums';
|
||||||
|
import {Connection, ConnectionBackend} from 'angular2/src/http/interfaces';
|
||||||
import * as Rx from 'rx';
|
import * as Rx from 'rx';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Connection represents a request and response for an underlying transport, like XHR or mock.
|
*
|
||||||
* The mock implementation contains helper methods to respond to connections within tests.
|
* Connection class used by MockBackend
|
||||||
* API subject to change and expand.
|
*
|
||||||
|
* This class is typically not instantiated directly, but instances can be retrieved by subscribing
|
||||||
|
*to the `connections` Observable of
|
||||||
|
* {@link MockBackend} in order to mock responses to requests.
|
||||||
|
*
|
||||||
**/
|
**/
|
||||||
export class Connection {
|
export class MockConnection implements Connection {
|
||||||
|
// TODO Name `readyState` should change to be more generic, and states could be made to be more
|
||||||
|
// descriptive than XHR states.
|
||||||
/**
|
/**
|
||||||
* Observer to call on download progress, if provided in config.
|
* Describes the state of the connection, based on `XMLHttpRequest.readyState`, but with
|
||||||
**/
|
* additional states. For example, state 5 indicates an aborted connection.
|
||||||
downloadObserver: Rx.Observer<Response>;
|
*/
|
||||||
|
|
||||||
/**
|
|
||||||
* TODO
|
|
||||||
* Name `readyState` should change to be more generic, and states could be made to be more
|
|
||||||
* descriptive than XHR states.
|
|
||||||
**/
|
|
||||||
|
|
||||||
readyState: ReadyStates;
|
readyState: ReadyStates;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link Request} instance used to create the connection.
|
||||||
|
*/
|
||||||
request: Request;
|
request: Request;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* [RxJS
|
||||||
|
* Observable](https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/core/observable.md)
|
||||||
|
* of {@link Response}. Can be subscribed to in order to be notified when a response is available.
|
||||||
|
*/
|
||||||
response: Rx.Subject<Response>;
|
response: Rx.Subject<Response>;
|
||||||
|
|
||||||
constructor(req: Request) {
|
constructor(req: Request) {
|
||||||
// State
|
|
||||||
if (Rx.hasOwnProperty('default')) {
|
if (Rx.hasOwnProperty('default')) {
|
||||||
this.response = new ((<any>Rx).default.Rx.Subject)();
|
this.response = new ((<any>Rx).default.Rx.Subject)();
|
||||||
} else {
|
} else {
|
||||||
|
@ -38,6 +47,9 @@ export class Connection {
|
||||||
this.dispose = this.dispose.bind(this);
|
this.dispose = this.dispose.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Changes the `readyState` of the connection to a custom state of 5 (cancelled).
|
||||||
|
*/
|
||||||
dispose() {
|
dispose() {
|
||||||
if (this.readyState !== ReadyStates.DONE) {
|
if (this.readyState !== ReadyStates.DONE) {
|
||||||
this.readyState = ReadyStates.CANCELLED;
|
this.readyState = ReadyStates.CANCELLED;
|
||||||
|
@ -45,8 +57,19 @@ export class Connection {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Called after a connection has been established.
|
* Sends a mock response to the connection. This response is the value that is emitted to the
|
||||||
**/
|
* `Observable` returned by {@link Http}.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* var connection;
|
||||||
|
* backend.connections.subscribe(c => connection = c);
|
||||||
|
* http.request('data.json').subscribe(res => console.log(res.text()));
|
||||||
|
* connection.mockRespond(new Response('fake response')); //logs 'fake response'
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
*/
|
||||||
mockRespond(res: Response) {
|
mockRespond(res: Response) {
|
||||||
if (this.readyState >= ReadyStates.DONE) {
|
if (this.readyState >= ReadyStates.DONE) {
|
||||||
throw new Error('Connection has already been resolved');
|
throw new Error('Connection has already been resolved');
|
||||||
|
@ -56,13 +79,24 @@ export class Connection {
|
||||||
this.response.onCompleted();
|
this.response.onCompleted();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Not yet implemented!
|
||||||
|
*
|
||||||
|
* Sends the provided {@link Response} to the `downloadObserver` of the `Request`
|
||||||
|
* associated with this connection.
|
||||||
|
*/
|
||||||
mockDownload(res: Response) {
|
mockDownload(res: Response) {
|
||||||
this.downloadObserver.onNext(res);
|
// this.request.downloadObserver.onNext(res);
|
||||||
if (res.bytesLoaded === res.totalBytes) {
|
// if (res.bytesLoaded === res.totalBytes) {
|
||||||
this.downloadObserver.onCompleted();
|
// this.request.downloadObserver.onCompleted();
|
||||||
}
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO(jeffbcross): consider using Response type
|
||||||
|
/**
|
||||||
|
* Emits the provided error object as an error to the {@link Response} observable returned
|
||||||
|
* from {@link Http}.
|
||||||
|
*/
|
||||||
mockError(err?) {
|
mockError(err?) {
|
||||||
// Matches XHR semantics
|
// Matches XHR semantics
|
||||||
this.readyState = ReadyStates.DONE;
|
this.readyState = ReadyStates.DONE;
|
||||||
|
@ -71,35 +105,135 @@ export class Connection {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A mock backend for testing the {@link Http} service.
|
||||||
|
*
|
||||||
|
* This class can be injected in tests, and should be used to override bindings
|
||||||
|
* to other backends, such as {@link XHRBackend}.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* import {MockBackend, DefaultOptions, Http} from 'angular2/http';
|
||||||
|
* it('should get some data', inject([AsyncTestCompleter], (async) => {
|
||||||
|
* var connection;
|
||||||
|
* var injector = Injector.resolveAndCreate([
|
||||||
|
* MockBackend,
|
||||||
|
* bind(Http).toFactory((backend, defaultOptions) => {
|
||||||
|
* return new Http(backend, defaultOptions)
|
||||||
|
* }, [MockBackend, DefaultOptions])]);
|
||||||
|
* var http = injector.get(Http);
|
||||||
|
* var backend = injector.get(MockBackend);
|
||||||
|
* //Assign any newly-created connection to local variable
|
||||||
|
* backend.connections.subscribe(c => connection = c);
|
||||||
|
* http.request('data.json').subscribe((res) => {
|
||||||
|
* expect(res.text()).toBe('awesome');
|
||||||
|
* async.done();
|
||||||
|
* });
|
||||||
|
* connection.mockRespond(new Response('awesome'));
|
||||||
|
* }));
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
* This method only exists in the mock implementation, not in real Backends.
|
||||||
|
**/
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class MockBackend {
|
export class MockBackend implements ConnectionBackend {
|
||||||
connections: Rx.Subject<Connection>;
|
/**
|
||||||
connectionsArray: Array<Connection>;
|
* [RxJS
|
||||||
pendingConnections: Rx.Observable<Connection>;
|
* Subject](https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/subjects/subject.md)
|
||||||
|
* of {@link MockConnection} instances that have been created by this backend. Can be subscribed
|
||||||
|
* to in order to respond to connections.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* import {MockBackend, Http, BaseRequestOptions} from 'angular2/http';
|
||||||
|
* import {Injector} from 'angular2/di';
|
||||||
|
*
|
||||||
|
* it('should get a response', () => {
|
||||||
|
* var connection; //this will be set when a new connection is emitted from the backend.
|
||||||
|
* var text; //this will be set from mock response
|
||||||
|
* var injector = Injector.resolveAndCreate([
|
||||||
|
* MockBackend,
|
||||||
|
* bind(Http).toFactory(backend, options) {
|
||||||
|
* return new Http(backend, options);
|
||||||
|
* }, [MockBackend, BaseRequestOptions]]);
|
||||||
|
* var backend = injector.get(MockBackend);
|
||||||
|
* var http = injector.get(Http);
|
||||||
|
* backend.connections.subscribe(c => connection = c);
|
||||||
|
* http.request('something.json').subscribe(res => {
|
||||||
|
* text = res.text();
|
||||||
|
* });
|
||||||
|
* connection.mockRespond(new Response('Something'));
|
||||||
|
* expect(text).toBe('Something');
|
||||||
|
* });
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
* This property only exists in the mock implementation, not in real Backends.
|
||||||
|
*/
|
||||||
|
connections: Rx.Subject<MockConnection>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An array representation of `connections`. This array will be updated with each connection that
|
||||||
|
* is created by this backend.
|
||||||
|
*
|
||||||
|
* This property only exists in the mock implementation, not in real Backends.
|
||||||
|
*/
|
||||||
|
connectionsArray: Array<MockConnection>;
|
||||||
|
/**
|
||||||
|
* [Observable](https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/core/observable.md)
|
||||||
|
* of {@link MockConnection} instances that haven't yet been resolved (i.e. with a `readyState`
|
||||||
|
* less than 4). Used internally to verify that no connections are pending via the
|
||||||
|
* `verifyNoPendingRequests` method.
|
||||||
|
*
|
||||||
|
* This property only exists in the mock implementation, not in real Backends.
|
||||||
|
*/
|
||||||
|
pendingConnections: Rx.Observable<MockConnection>;
|
||||||
constructor() {
|
constructor() {
|
||||||
|
var Observable;
|
||||||
this.connectionsArray = [];
|
this.connectionsArray = [];
|
||||||
if (Rx.hasOwnProperty('default')) {
|
if (Rx.hasOwnProperty('default')) {
|
||||||
this.connections = new (<any>Rx).default.Rx.Subject();
|
this.connections = new (<any>Rx).default.Rx.Subject();
|
||||||
|
Observable = (<any>Rx).default.Rx.Observable;
|
||||||
} else {
|
} else {
|
||||||
this.connections = new Rx.Subject<Connection>();
|
this.connections = new Rx.Subject<MockConnection>();
|
||||||
|
Observable = Rx.Observable;
|
||||||
}
|
}
|
||||||
this.connections.subscribe(connection => this.connectionsArray.push(connection));
|
this.connections.subscribe(connection => this.connectionsArray.push(connection));
|
||||||
this.pendingConnections = this.connections.filter((c) => c.readyState < ReadyStates.DONE);
|
this.pendingConnections =
|
||||||
|
Observable.fromArray(this.connectionsArray).filter((c) => c.readyState < ReadyStates.DONE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks all connections, and raises an exception if any connection has not received a response.
|
||||||
|
*
|
||||||
|
* This method only exists in the mock implementation, not in real Backends.
|
||||||
|
*/
|
||||||
verifyNoPendingRequests() {
|
verifyNoPendingRequests() {
|
||||||
let pending = 0;
|
let pending = 0;
|
||||||
this.pendingConnections.subscribe((c) => pending++);
|
this.pendingConnections.subscribe((c) => pending++);
|
||||||
if (pending > 0) throw new Error(`${pending} pending connections to be resolved`);
|
if (pending > 0) throw new Error(`${pending} pending connections to be resolved`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Can be used in conjunction with `verifyNoPendingRequests` to resolve any not-yet-resolve
|
||||||
|
* connections, if it's expected that there are connections that have not yet received a response.
|
||||||
|
*
|
||||||
|
* This method only exists in the mock implementation, not in real Backends.
|
||||||
|
*/
|
||||||
resolveAllConnections() { this.connections.subscribe((c) => c.readyState = 4); }
|
resolveAllConnections() { this.connections.subscribe((c) => c.readyState = 4); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new {@link MockConnection}. This is equivalent to calling `new
|
||||||
|
* MockConnection()`, except that it also will emit the new `Connection` to the `connections`
|
||||||
|
* observable of this `MockBackend` instance. This method will usually only be used by tests
|
||||||
|
* against the framework itself, not by end-users.
|
||||||
|
*/
|
||||||
createConnection(req: Request) {
|
createConnection(req: Request) {
|
||||||
if (!req || !(req instanceof Request)) {
|
if (!req || !(req instanceof Request)) {
|
||||||
throw new Error(`createConnection requires an instance of Request, got ${req}`);
|
throw new Error(`createConnection requires an instance of Request, got ${req}`);
|
||||||
}
|
}
|
||||||
let connection = new Connection(req);
|
let connection = new MockConnection(req);
|
||||||
this.connections.onNext(connection);
|
this.connections.onNext(connection);
|
||||||
return connection;
|
return connection;
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,8 +7,21 @@ import {Injectable} from 'angular2/di';
|
||||||
import {BrowserXHR} from './browser_xhr';
|
import {BrowserXHR} from './browser_xhr';
|
||||||
import * as Rx from 'rx';
|
import * as Rx from 'rx';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates connections using `XMLHttpRequest`. Given a fully-qualified
|
||||||
|
* request, an `XHRConnection` will immediately create an `XMLHttpRequest` object and send the
|
||||||
|
* request.
|
||||||
|
*
|
||||||
|
* This class would typically not be created or interacted with directly inside applications, though
|
||||||
|
* the {@link MockConnection} may be interacted with in tests.
|
||||||
|
*/
|
||||||
export class XHRConnection implements Connection {
|
export class XHRConnection implements Connection {
|
||||||
request: Request;
|
request: Request;
|
||||||
|
/**
|
||||||
|
* Response
|
||||||
|
* [Subject](https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/subjects/subject.md)
|
||||||
|
* which emits a single {@link Response} value on load event of `XMLHttpRequest`.
|
||||||
|
*/
|
||||||
response: Rx.Subject<Response>;
|
response: Rx.Subject<Response>;
|
||||||
readyState: ReadyStates;
|
readyState: ReadyStates;
|
||||||
private _xhr;
|
private _xhr;
|
||||||
|
@ -20,6 +33,7 @@ export class XHRConnection implements Connection {
|
||||||
this.response = new Rx.Subject<Response>();
|
this.response = new Rx.Subject<Response>();
|
||||||
}
|
}
|
||||||
this._xhr = new NativeConstruct();
|
this._xhr = new NativeConstruct();
|
||||||
|
// TODO(jeffbcross): implement error listening/propagation
|
||||||
this._xhr.open(RequestMethods[req.method], req.url);
|
this._xhr.open(RequestMethods[req.method], req.url);
|
||||||
this._xhr.addEventListener(
|
this._xhr.addEventListener(
|
||||||
'load',
|
'load',
|
||||||
|
@ -28,9 +42,38 @@ export class XHRConnection implements Connection {
|
||||||
this._xhr.send(this.request.text());
|
this._xhr.send(this.request.text());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calls abort on the underlying XMLHttpRequest.
|
||||||
|
*/
|
||||||
dispose(): void { this._xhr.abort(); }
|
dispose(): void { this._xhr.abort(); }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates {@link XHRConnection} instances.
|
||||||
|
*
|
||||||
|
* This class would typically not be used by end users, but could be
|
||||||
|
* overridden if a different backend implementation should be used,
|
||||||
|
* such as in a node backend.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* import {Http, MyNodeBackend, httpInjectables, BaseRequestOptions} from 'angular2/http';
|
||||||
|
* @Component({
|
||||||
|
* appInjector: [
|
||||||
|
* httpInjectables,
|
||||||
|
* bind(Http).toFactory((backend, options) => {
|
||||||
|
* return new Http(backend, options);
|
||||||
|
* }, [MyNodeBackend, BaseRequestOptions])]
|
||||||
|
* })
|
||||||
|
* class MyComponent {
|
||||||
|
* constructor(http:Http) {
|
||||||
|
* http('people.json').subscribe(res => this.people = res.json());
|
||||||
|
* }
|
||||||
|
* }
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
**/
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class XHRBackend implements ConnectionBackend {
|
export class XHRBackend implements ConnectionBackend {
|
||||||
constructor(private _NativeConstruct: BrowserXHR) {}
|
constructor(private _NativeConstruct: BrowserXHR) {}
|
||||||
|
|
|
@ -6,9 +6,28 @@ import {IRequestOptions} from './interfaces';
|
||||||
import {Injectable} from 'angular2/di';
|
import {Injectable} from 'angular2/di';
|
||||||
import {ListWrapper, StringMapWrapper} from 'angular2/src/facade/collection';
|
import {ListWrapper, StringMapWrapper} from 'angular2/src/facade/collection';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a request options object with default properties as described in the [Fetch
|
||||||
|
* Spec](https://fetch.spec.whatwg.org/#requestinit) to be optionally provided when instantiating a
|
||||||
|
* {@link Request}. This class is used implicitly by {@link Http} to merge in provided request
|
||||||
|
* options with the default options specified here. These same default options are injectable via
|
||||||
|
* the {@link BaseRequestOptions} class.
|
||||||
|
*/
|
||||||
export class RequestOptions implements IRequestOptions {
|
export class RequestOptions implements IRequestOptions {
|
||||||
|
/**
|
||||||
|
* Http method with which to execute the request.
|
||||||
|
*
|
||||||
|
* Defaults to "GET".
|
||||||
|
*/
|
||||||
method: RequestMethods = RequestMethods.GET;
|
method: RequestMethods = RequestMethods.GET;
|
||||||
|
/**
|
||||||
|
* Headers object based on the `Headers` class in the [Fetch
|
||||||
|
* Spec](https://fetch.spec.whatwg.org/#headers-class).
|
||||||
|
*/
|
||||||
headers: Headers;
|
headers: Headers;
|
||||||
|
/**
|
||||||
|
* Body to be used when creating the request.
|
||||||
|
*/
|
||||||
body: URLSearchParams | FormData | Blob | string;
|
body: URLSearchParams | FormData | Blob | string;
|
||||||
mode: RequestModesOpts = RequestModesOpts.Cors;
|
mode: RequestModesOpts = RequestModesOpts.Cors;
|
||||||
credentials: RequestCredentialsOpts;
|
credentials: RequestCredentialsOpts;
|
||||||
|
@ -25,11 +44,33 @@ export class RequestOptions implements IRequestOptions {
|
||||||
this.cache = cache;
|
this.cache = cache;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a copy of the `RequestOptions` instance, using the optional input as values to override
|
||||||
|
* existing values.
|
||||||
|
*/
|
||||||
merge(opts: IRequestOptions = {}): RequestOptions {
|
merge(opts: IRequestOptions = {}): RequestOptions {
|
||||||
return new RequestOptions(StringMapWrapper.merge(this, opts));
|
return new RequestOptions(StringMapWrapper.merge(this, opts));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injectable version of {@link RequestOptions}.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* import {Http, BaseRequestOptions, Request} from 'angular2/http';
|
||||||
|
* ...
|
||||||
|
* class MyComponent {
|
||||||
|
* constructor(baseRequestOptions:BaseRequestOptions, http:Http) {
|
||||||
|
* var options = baseRequestOptions.merge({body: 'foobar'});
|
||||||
|
* var request = new Request('https://foo', options);
|
||||||
|
* http.request(request).subscribe(res => this.bars = res.json());
|
||||||
|
* }
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
*/
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class BaseRequestOptions extends RequestOptions {
|
export class BaseRequestOptions extends RequestOptions {
|
||||||
constructor() { super(); }
|
constructor() { super(); }
|
||||||
|
|
|
@ -14,9 +14,11 @@ import {
|
||||||
ListWrapper
|
ListWrapper
|
||||||
} from 'angular2/src/facade/collection';
|
} from 'angular2/src/facade/collection';
|
||||||
|
|
||||||
// (@jeffbcross): This is implemented mostly to spec, except that the entries method has been
|
/**
|
||||||
// removed because it doesn't exist in dart, and it doesn't seem worth adding it to the facade.
|
* Polyfill for [Headers](https://developer.mozilla.org/en-US/docs/Web/API/Headers/Headers), as
|
||||||
|
* specified in the [Fetch Spec](https://fetch.spec.whatwg.org/#headers-class). The only known
|
||||||
|
* difference from the spec is the lack of an `entries` method.
|
||||||
|
*/
|
||||||
export class Headers {
|
export class Headers {
|
||||||
_headersMap: Map<string, List<string>>;
|
_headersMap: Map<string, List<string>>;
|
||||||
constructor(headers?: Headers | Object) {
|
constructor(headers?: Headers | Object) {
|
||||||
|
|
|
@ -10,41 +10,6 @@ import {RequestMethods} from './enums';
|
||||||
import {URLSearchParams} from './url_search_params';
|
import {URLSearchParams} from './url_search_params';
|
||||||
import * as Rx from 'rx';
|
import * as Rx from 'rx';
|
||||||
|
|
||||||
/**
|
|
||||||
* A function to perform http requests over XMLHttpRequest.
|
|
||||||
*
|
|
||||||
* #Example
|
|
||||||
*
|
|
||||||
* ```
|
|
||||||
* @Component({
|
|
||||||
* appInjector: [httpBindings]
|
|
||||||
* })
|
|
||||||
* @View({
|
|
||||||
* directives: [NgFor],
|
|
||||||
* template: `
|
|
||||||
* <ul>
|
|
||||||
* <li *ng-for="#person of people">
|
|
||||||
* hello, {{person.name}}
|
|
||||||
* </li>
|
|
||||||
* </ul>
|
|
||||||
* `
|
|
||||||
* })
|
|
||||||
* class MyComponent {
|
|
||||||
* constructor(http:Http) {
|
|
||||||
* http('people.json').subscribe(res => this.people = res.json());
|
|
||||||
* }
|
|
||||||
* }
|
|
||||||
* ```
|
|
||||||
*
|
|
||||||
*
|
|
||||||
* This function is bound to a single underlying connection mechanism, such as XHR, which could be
|
|
||||||
* mocked with dependency injection by replacing the `Backend` binding. For other transports, like
|
|
||||||
* JSONP or Node, a separate http function would be created, such as httpJSONP.
|
|
||||||
*
|
|
||||||
* @exportedAs angular2/http
|
|
||||||
*
|
|
||||||
**/
|
|
||||||
|
|
||||||
function httpRequest(backend: XHRBackend, request: Request) {
|
function httpRequest(backend: XHRBackend, request: Request) {
|
||||||
return <Rx.Observable<Response>>(Observable.create(observer => {
|
return <Rx.Observable<Response>>(Observable.create(observer => {
|
||||||
var connection: Connection = backend.createConnection(request);
|
var connection: Connection = backend.createConnection(request);
|
||||||
|
@ -56,51 +21,123 @@ function httpRequest(backend: XHRBackend, request: Request) {
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Abstract
|
/**
|
||||||
|
* Performs http requests using `XMLHttpRequest` as the default backend.
|
||||||
|
*
|
||||||
|
* `Http` is available as an injectable class, with methods to perform http requests. Calling
|
||||||
|
* `request` returns an
|
||||||
|
* [Observable](https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/core/observable.md),
|
||||||
|
* which will emit a single {@link Response} when a response is
|
||||||
|
* received.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* import {Http, httpInjectables} from 'angular2/http';
|
||||||
|
* @Component({selector: 'http-app', appInjector: [httpInjectables]})
|
||||||
|
* @View({templateUrl: 'people.html'})
|
||||||
|
* class PeopleComponent {
|
||||||
|
* constructor(http: Http) {
|
||||||
|
* http('people.json')
|
||||||
|
* // Call map on the response observable to get the parsed people object
|
||||||
|
* .map(res => res.json())
|
||||||
|
* // Subscribe to the observable to get the parsed people object and attach it to the
|
||||||
|
* // component
|
||||||
|
* .subscribe(people => this.people = people);
|
||||||
|
* }
|
||||||
|
* }
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
* The default construct used to perform requests, `XMLHttpRequest`, is abstracted as a "Backend" (
|
||||||
|
* {@link XHRBackend} in this case), which could be mocked with dependency injection by replacing
|
||||||
|
* the {@link XHRBackend} binding, as in the following example:
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* import {MockBackend, BaseRequestOptions, Http} from 'angular2/http';
|
||||||
|
* var injector = Injector.resolveAndCreate([
|
||||||
|
* BaseRequestOptions,
|
||||||
|
* MockBackend,
|
||||||
|
* bind(Http).toFactory(
|
||||||
|
* function(backend, defaultOptions) {
|
||||||
|
* return new Http(backend, defaultOptions);
|
||||||
|
* },
|
||||||
|
* [MockBackend, BaseRequestOptions])
|
||||||
|
* ]);
|
||||||
|
* var http = injector.get(Http);
|
||||||
|
* http.get('request-from-mock-backend.json').subscribe((res:Response) => doSomething(res));
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
**/
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class Http {
|
export class Http {
|
||||||
constructor(private backend: XHRBackend, private defaultOptions: BaseRequestOptions) {}
|
constructor(private _backend: XHRBackend, private _defaultOptions: BaseRequestOptions) {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performs any type of http request. First argument is required, and can either be a url or
|
||||||
|
* a {@link Request} instance. If the first argument is a url, an optional {@link RequestOptions}
|
||||||
|
* object can be provided as the 2nd argument. The options object will be merged with the values
|
||||||
|
* of {@link BaseRequestOptions} before performing the request.
|
||||||
|
*/
|
||||||
request(url: string | Request, options?: IRequestOptions): Rx.Observable<Response> {
|
request(url: string | Request, options?: IRequestOptions): Rx.Observable<Response> {
|
||||||
if (typeof url === 'string') {
|
if (typeof url === 'string') {
|
||||||
return httpRequest(this.backend, new Request(url, this.defaultOptions.merge(options)));
|
return httpRequest(this._backend, new Request(url, this._defaultOptions.merge(options)));
|
||||||
} else if (url instanceof Request) {
|
} else if (url instanceof Request) {
|
||||||
return httpRequest(this.backend, url);
|
return httpRequest(this._backend, url);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performs a request with `get` http method.
|
||||||
|
*/
|
||||||
get(url: string, options?: IRequestOptions) {
|
get(url: string, options?: IRequestOptions) {
|
||||||
return httpRequest(this.backend, new Request(url, this.defaultOptions.merge(options)
|
return httpRequest(this._backend, new Request(url, this._defaultOptions.merge(options)
|
||||||
.merge({method: RequestMethods.GET})));
|
.merge({method: RequestMethods.GET})));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performs a request with `post` http method.
|
||||||
|
*/
|
||||||
post(url: string, body: URLSearchParams | FormData | Blob | string, options?: IRequestOptions) {
|
post(url: string, body: URLSearchParams | FormData | Blob | string, options?: IRequestOptions) {
|
||||||
return httpRequest(this.backend,
|
return httpRequest(this._backend,
|
||||||
new Request(url, this.defaultOptions.merge(options)
|
new Request(url, this._defaultOptions.merge(options)
|
||||||
|
|
||||||
.merge({body: body, method: RequestMethods.POST})));
|
.merge({body: body, method: RequestMethods.POST})));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performs a request with `put` http method.
|
||||||
|
*/
|
||||||
put(url: string, body: URLSearchParams | FormData | Blob | string, options?: IRequestOptions) {
|
put(url: string, body: URLSearchParams | FormData | Blob | string, options?: IRequestOptions) {
|
||||||
return httpRequest(this.backend,
|
return httpRequest(this._backend,
|
||||||
new Request(url, this.defaultOptions.merge(options)
|
new Request(url, this._defaultOptions.merge(options)
|
||||||
.merge({body: body, method: RequestMethods.PUT})));
|
.merge({body: body, method: RequestMethods.PUT})));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performs a request with `delete` http method.
|
||||||
|
*/
|
||||||
delete (url: string, options?: IRequestOptions) {
|
delete (url: string, options?: IRequestOptions) {
|
||||||
return httpRequest(this.backend, new Request(url, this.defaultOptions.merge(options)
|
return httpRequest(this._backend, new Request(url, this._defaultOptions.merge(options).merge(
|
||||||
.merge({method: RequestMethods.DELETE})));
|
{method: RequestMethods.DELETE})));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performs a request with `patch` http method.
|
||||||
|
*/
|
||||||
patch(url: string, body: URLSearchParams | FormData | Blob | string, options?: IRequestOptions) {
|
patch(url: string, body: URLSearchParams | FormData | Blob | string, options?: IRequestOptions) {
|
||||||
return httpRequest(this.backend,
|
return httpRequest(this._backend,
|
||||||
new Request(url, this.defaultOptions.merge(options)
|
new Request(url, this._defaultOptions.merge(options)
|
||||||
.merge({body: body, method: RequestMethods.PATCH})));
|
.merge({body: body, method: RequestMethods.PATCH})));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performs a request with `head` http method.
|
||||||
|
*/
|
||||||
head(url: string, options?: IRequestOptions) {
|
head(url: string, options?: IRequestOptions) {
|
||||||
return httpRequest(this.backend, new Request(url, this.defaultOptions.merge(options)
|
return httpRequest(this._backend, new Request(url, this._defaultOptions.merge(options)
|
||||||
.merge({method: RequestMethods.HEAD})));
|
.merge({method: RequestMethods.HEAD})));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -110,6 +147,30 @@ if (Rx.hasOwnProperty('default')) {
|
||||||
} else {
|
} else {
|
||||||
Observable = Rx.Observable;
|
Observable = Rx.Observable;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* Alias to the `request` method of {@link Http}, for those who'd prefer a simple function instead
|
||||||
|
* of an object. In order to get TypeScript type information about the `HttpFactory`, the {@link
|
||||||
|
* IHttp} interface can be used as shown in the following example.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* import {httpInjectables, HttpFactory, IHttp} from 'angular2/http';
|
||||||
|
* @Component({
|
||||||
|
* appInjector: [httpInjectables]
|
||||||
|
* })
|
||||||
|
* @View({
|
||||||
|
* templateUrl: 'people.html'
|
||||||
|
* })
|
||||||
|
* class MyComponent {
|
||||||
|
* constructor(@Inject(HttpFactory) http:IHttp) {
|
||||||
|
* http('people.json').subscribe(res => this.people = res.json());
|
||||||
|
* }
|
||||||
|
* }
|
||||||
|
* ```
|
||||||
|
**/
|
||||||
export function HttpFactory(backend: XHRBackend, defaultOptions: BaseRequestOptions) {
|
export function HttpFactory(backend: XHRBackend, defaultOptions: BaseRequestOptions) {
|
||||||
return function(url: string | Request, options?: IRequestOptions) {
|
return function(url: string | Request, options?: IRequestOptions) {
|
||||||
if (typeof url === 'string') {
|
if (typeof url === 'string') {
|
||||||
|
|
|
@ -58,6 +58,27 @@ export interface Connection {
|
||||||
dispose(): void;
|
dispose(): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides an interface to provide type information for {@link HttpFactory} when injecting.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* * import {httpInjectables, HttpFactory, IHttp} from 'angular2/http';
|
||||||
|
* @Component({
|
||||||
|
* appInjector: [httpInjectables]
|
||||||
|
* })
|
||||||
|
* @View({
|
||||||
|
* templateUrl: 'people.html'
|
||||||
|
* })
|
||||||
|
* class MyComponent {
|
||||||
|
* constructor(@Inject(HttpFactory) http:IHttp) {
|
||||||
|
* http('people.json').subscribe(res => this.people = res.json());
|
||||||
|
* }
|
||||||
|
* }
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
*/
|
||||||
// Prefixed as IHttp because used in conjunction with Http class, but interface is callable
|
// Prefixed as IHttp because used in conjunction with Http class, but interface is callable
|
||||||
// constructor(@Inject(Http) http:IHttp)
|
// constructor(@Inject(Http) http:IHttp)
|
||||||
export interface IHttp { (url: string, options?: IRequestOptions): Rx.Observable<Response> }
|
export interface IHttp { (url: string, options?: IRequestOptions): Rx.Observable<Response> }
|
||||||
|
|
|
@ -4,26 +4,37 @@ import {IRequestOptions, Request as IRequest} from './interfaces';
|
||||||
import {Headers} from './headers';
|
import {Headers} from './headers';
|
||||||
import {BaseException, RegExpWrapper} from 'angular2/src/facade/lang';
|
import {BaseException, RegExpWrapper} from 'angular2/src/facade/lang';
|
||||||
|
|
||||||
// TODO(jeffbcross): implement body accessors
|
// TODO(jeffbcross): properly implement body accessors
|
||||||
|
/**
|
||||||
|
* Creates `Request` instances with default values.
|
||||||
|
*
|
||||||
|
* The Request's interface is inspired by the Request constructor defined in the [Fetch
|
||||||
|
* Spec](https://fetch.spec.whatwg.org/#request-class),
|
||||||
|
* but is considered a static value whose body can be accessed many times. There are other
|
||||||
|
* differences in the implementation, but this is the most significant.
|
||||||
|
*/
|
||||||
export class Request implements IRequest {
|
export class Request implements IRequest {
|
||||||
|
/**
|
||||||
|
* Http method with which to perform the request.
|
||||||
|
*
|
||||||
|
* Defaults to GET.
|
||||||
|
*/
|
||||||
method: RequestMethods;
|
method: RequestMethods;
|
||||||
mode: RequestModesOpts;
|
mode: RequestModesOpts;
|
||||||
credentials: RequestCredentialsOpts;
|
credentials: RequestCredentialsOpts;
|
||||||
headers: Headers;
|
/**
|
||||||
/*
|
* Headers object based on the `Headers` class in the [Fetch
|
||||||
* Non-Standard Properties
|
* Spec](https://fetch.spec.whatwg.org/#headers-class). {@link Headers} class reference.
|
||||||
*/
|
*/
|
||||||
// This property deviates from the standard. Body can be set in constructor, but is only
|
headers: Headers;
|
||||||
// accessible
|
|
||||||
// via json(), text(), arrayBuffer(), and blob() accessors, which also change the request's state
|
|
||||||
// to "used".
|
|
||||||
private body: URLSearchParams | FormData | Blob | string;
|
|
||||||
|
|
||||||
constructor(public url: string, {body, method = RequestMethods.GET, mode = RequestModesOpts.Cors,
|
private _body: URLSearchParams | FormData | Blob | string;
|
||||||
credentials = RequestCredentialsOpts.Omit,
|
|
||||||
headers = new Headers()}: IRequestOptions = {}) {
|
constructor(/** Url of the remote resource */ public url: string,
|
||||||
this.body = body;
|
{body, method = RequestMethods.GET, mode = RequestModesOpts.Cors,
|
||||||
// Defaults to 'GET', consistent with browser
|
credentials = RequestCredentialsOpts.Omit,
|
||||||
|
headers = new Headers()}: IRequestOptions = {}) {
|
||||||
|
this._body = body;
|
||||||
this.method = method;
|
this.method = method;
|
||||||
// Defaults to 'cors', consistent with browser
|
// Defaults to 'cors', consistent with browser
|
||||||
// TODO(jeffbcross): implement behavior
|
// TODO(jeffbcross): implement behavior
|
||||||
|
@ -31,9 +42,13 @@ export class Request implements IRequest {
|
||||||
// Defaults to 'omit', consistent with browser
|
// Defaults to 'omit', consistent with browser
|
||||||
// TODO(jeffbcross): implement behavior
|
// TODO(jeffbcross): implement behavior
|
||||||
this.credentials = credentials;
|
this.credentials = credentials;
|
||||||
// Defaults to empty headers object, consistent with browser
|
|
||||||
this.headers = headers;
|
this.headers = headers;
|
||||||
}
|
}
|
||||||
|
|
||||||
text(): String { return this.body ? this.body.toString() : ''; }
|
/**
|
||||||
|
* Returns the request's body as string, assuming that body exists. If body is undefined, return
|
||||||
|
* empty
|
||||||
|
* string.
|
||||||
|
*/
|
||||||
|
text(): String { return this._body ? this._body.toString() : ''; }
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,22 +4,79 @@ import {baseResponseOptions} from './base_response_options';
|
||||||
import {BaseException, isJsObject, isString, global} from 'angular2/src/facade/lang';
|
import {BaseException, isJsObject, isString, global} from 'angular2/src/facade/lang';
|
||||||
import {Headers} from './headers';
|
import {Headers} from './headers';
|
||||||
|
|
||||||
// TODO: make this injectable so baseResponseOptions can be overridden
|
// TODO: make this injectable so baseResponseOptions can be overridden, mostly for the benefit of
|
||||||
|
// headers merging.
|
||||||
|
/**
|
||||||
|
* Creates `Response` instances with default values.
|
||||||
|
*
|
||||||
|
* Though this object isn't
|
||||||
|
* usually instantiated by end-users, it is the primary object interacted with when it comes time to
|
||||||
|
* add data to a view.
|
||||||
|
*
|
||||||
|
* #Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* http.request('my-friends.txt').subscribe(response => this.friends = response.text());
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
* The Response's interface is inspired by the Request constructor defined in the [Fetch
|
||||||
|
* Spec](https://fetch.spec.whatwg.org/#response-class), but is considered a static value whose body
|
||||||
|
* can be accessed many times. There are other differences in the implementation, but this is the
|
||||||
|
* most significant.
|
||||||
|
*/
|
||||||
export class Response implements IResponse {
|
export class Response implements IResponse {
|
||||||
|
/**
|
||||||
|
* One of "basic", "cors", "default", "error, or "opaque".
|
||||||
|
*
|
||||||
|
* Defaults to "default".
|
||||||
|
*/
|
||||||
type: ResponseTypes;
|
type: ResponseTypes;
|
||||||
|
/**
|
||||||
|
* True if the response's status is within 200-299
|
||||||
|
*/
|
||||||
ok: boolean;
|
ok: boolean;
|
||||||
|
/**
|
||||||
|
* URL of response.
|
||||||
|
*
|
||||||
|
* Defaults to empty string.
|
||||||
|
*/
|
||||||
url: string;
|
url: string;
|
||||||
|
/**
|
||||||
|
* Status code returned by server.
|
||||||
|
*
|
||||||
|
* Defaults to 200.
|
||||||
|
*/
|
||||||
status: number;
|
status: number;
|
||||||
|
/**
|
||||||
|
* Text representing the corresponding reason phrase to the `status`, as defined in [ietf rfc 2616
|
||||||
|
* section 6.1.1](https://tools.ietf.org/html/rfc2616#section-6.1.1)
|
||||||
|
*
|
||||||
|
* Defaults to "OK"
|
||||||
|
*/
|
||||||
statusText: string;
|
statusText: string;
|
||||||
|
/**
|
||||||
|
* Non-standard property
|
||||||
|
*
|
||||||
|
* Denotes how many of the response body's bytes have been loaded, for example if the response is
|
||||||
|
* the result of a progress event.
|
||||||
|
*/
|
||||||
bytesLoaded: number;
|
bytesLoaded: number;
|
||||||
|
/**
|
||||||
|
* Non-standard property
|
||||||
|
*
|
||||||
|
* Denotes how many bytes are expected in the final response body.
|
||||||
|
*/
|
||||||
totalBytes: number;
|
totalBytes: number;
|
||||||
|
/**
|
||||||
|
* Headers object based on the `Headers` class in the [Fetch
|
||||||
|
* Spec](https://fetch.spec.whatwg.org/#headers-class).
|
||||||
|
*/
|
||||||
headers: Headers;
|
headers: Headers;
|
||||||
constructor(private body?: string | Object | ArrayBuffer | JSON | FormData | Blob,
|
constructor(private _body?: string | Object | ArrayBuffer | JSON | FormData | Blob,
|
||||||
{status, statusText, headers, type, url}: ResponseOptions = baseResponseOptions) {
|
{status, statusText, headers, type, url}: ResponseOptions = baseResponseOptions) {
|
||||||
if (isJsObject(headers)) {
|
if (isJsObject(headers)) {
|
||||||
headers = new Headers(headers);
|
headers = new Headers(headers);
|
||||||
}
|
}
|
||||||
this.body = body;
|
|
||||||
this.status = status;
|
this.status = status;
|
||||||
this.statusText = statusText;
|
this.statusText = statusText;
|
||||||
this.headers = <Headers>headers;
|
this.headers = <Headers>headers;
|
||||||
|
@ -27,20 +84,32 @@ export class Response implements IResponse {
|
||||||
this.url = url;
|
this.url = url;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Not yet implemented
|
||||||
|
*/
|
||||||
blob(): Blob {
|
blob(): Blob {
|
||||||
throw new BaseException('"blob()" method not implemented on Response superclass');
|
throw new BaseException('"blob()" method not implemented on Response superclass');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to return body as parsed `JSON` object, or raises an exception.
|
||||||
|
*/
|
||||||
json(): JSON {
|
json(): JSON {
|
||||||
if (isJsObject(this.body)) {
|
if (isJsObject(this._body)) {
|
||||||
return <JSON>this.body;
|
return <JSON>this._body;
|
||||||
} else if (isString(this.body)) {
|
} else if (isString(this._body)) {
|
||||||
return global.JSON.parse(<string>this.body);
|
return global.JSON.parse(<string>this._body);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
text(): string { return this.body.toString(); }
|
/**
|
||||||
|
* Returns the body as a string, presuming `toString()` can be called on the response body.
|
||||||
|
*/
|
||||||
|
text(): string { return this._body.toString(); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Not yet implemented
|
||||||
|
*/
|
||||||
arrayBuffer(): ArrayBuffer {
|
arrayBuffer(): ArrayBuffer {
|
||||||
throw new BaseException('"arrayBuffer()" method not implemented on Response superclass');
|
throw new BaseException('"arrayBuffer()" method not implemented on Response superclass');
|
||||||
}
|
}
|
||||||
|
|
|
@ -80,9 +80,13 @@ export function main() {
|
||||||
connection.mockRespond(baseResponse)
|
connection.mockRespond(baseResponse)
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
|
||||||
it('should accept a fully-qualified request as its only parameter', () => {
|
it('should accept a fully-qualified request as its only parameter', () => {
|
||||||
var req = new Request('https://google.com');
|
var req = new Request('https://google.com');
|
||||||
backend.connections.subscribe(c => { expect(c.request.url).toBe('https://google.com'); });
|
backend.connections.subscribe(c => {
|
||||||
|
expect(c.request.url).toBe('https://google.com');
|
||||||
|
c.mockRespond(new Response('Thank you'));
|
||||||
|
});
|
||||||
httpFactory(req).subscribe(() => {});
|
httpFactory(req).subscribe(() => {});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -116,15 +120,16 @@ export function main() {
|
||||||
|
|
||||||
describe('Http', () => {
|
describe('Http', () => {
|
||||||
describe('.request()', () => {
|
describe('.request()', () => {
|
||||||
it('should return an Observable', () => {
|
it('should return an Observable',
|
||||||
expect(typeof http.request(url).subscribe).toBe('function');
|
() => { expect(typeof http.request(url).subscribe).toBe('function'); });
|
||||||
backend.resolveAllConnections();
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
it('should accept a fully-qualified request as its only parameter', () => {
|
it('should accept a fully-qualified request as its only parameter', () => {
|
||||||
var req = new Request('https://google.com');
|
var req = new Request('https://google.com');
|
||||||
backend.connections.subscribe(c => { expect(c.request.url).toBe('https://google.com'); });
|
backend.connections.subscribe(c => {
|
||||||
|
expect(c.request.url).toBe('https://google.com');
|
||||||
|
c.mockRespond(new Response('Thank you'));
|
||||||
|
});
|
||||||
http.request(req).subscribe(() => {});
|
http.request(req).subscribe(() => {});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,20 +0,0 @@
|
||||||
import {Directive, ViewContainerRef, ProtoViewRef} from "angular2/angular2";
|
|
||||||
|
|
||||||
@Directive({selector: '[assign-local]', properties: ['localVariable: assignLocalTo']})
|
|
||||||
export class LocalVariable {
|
|
||||||
viewContainer: ViewContainerRef;
|
|
||||||
protoViewRef: ProtoViewRef;
|
|
||||||
view: any;
|
|
||||||
constructor(viewContainer: ViewContainerRef, protoViewRef: ProtoViewRef) {
|
|
||||||
this.viewContainer = viewContainer;
|
|
||||||
this.protoViewRef = protoViewRef;
|
|
||||||
}
|
|
||||||
|
|
||||||
set localVariable(exp) {
|
|
||||||
if (!this.viewContainer.length) {
|
|
||||||
this.view = this.viewContainer.create(this.protoViewRef);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.view.setLocal("$implicit", exp);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,30 +1,21 @@
|
||||||
import {bootstrap, Component, View, NgFor, NgIf, Inject} from 'angular2/angular2';
|
import {bootstrap, Component, View, NgFor, Inject} from 'angular2/angular2';
|
||||||
import {httpInjectables} from 'angular2/http';
|
import {Http, httpInjectables} from 'angular2/http';
|
||||||
import {HttpFactory} from 'angular2/src/http/http';
|
|
||||||
import {IHttp} from 'angular2/src/http/interfaces';
|
|
||||||
import {Response} from 'angular2/src/http/static_response';
|
|
||||||
import {LocalVariable} from './assign_local_directive';
|
|
||||||
|
|
||||||
@Component({selector: 'http-app', appInjector: [httpInjectables]})
|
@Component({selector: 'http-app'})
|
||||||
@View({
|
@View({
|
||||||
directives: [NgFor, NgIf, LocalVariable],
|
directives: [NgFor],
|
||||||
template: `
|
template: `
|
||||||
<h1>people</h1>
|
<h1>people</h1>
|
||||||
<div *assign-local="#unwrappedPeople to people | rx">
|
<ul class="people">
|
||||||
<ul *ng-if="unwrappedPeople" class="people">
|
<li *ng-for="#person of people">
|
||||||
<li *ng-for="#person of unwrappedPeople">
|
hello, {{person.name}}
|
||||||
hello, {{person.name}}
|
</li>
|
||||||
</li>
|
</ul>
|
||||||
</ul>
|
|
||||||
<span *ng-if="!unwrappedPeople">
|
|
||||||
Fetching people...
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
export class HttpCmp {
|
export class HttpCmp {
|
||||||
people: Rx.Observable<Object>;
|
people: Object;
|
||||||
constructor(@Inject(HttpFactory) http: IHttp) {
|
constructor(http: Http) {
|
||||||
this.people = http('./people.json').map(res => res.json());
|
http.get('./people.json').map(res => res.json()).subscribe(people => this.people = people);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -2,29 +2,13 @@
|
||||||
|
|
||||||
import {
|
import {
|
||||||
bootstrap,
|
bootstrap,
|
||||||
ElementRef,
|
|
||||||
Component,
|
|
||||||
Directive,
|
|
||||||
View,
|
|
||||||
Injectable,
|
|
||||||
NgFor,
|
|
||||||
NgIf,
|
|
||||||
Inject
|
|
||||||
} from 'angular2/angular2';
|
} from 'angular2/angular2';
|
||||||
import {bind} from 'angular2/di';
|
|
||||||
import {PipeRegistry, defaultPipes} from 'angular2/change_detection';
|
|
||||||
import {reflector} from 'angular2/src/reflection/reflection';
|
import {reflector} from 'angular2/src/reflection/reflection';
|
||||||
import {ReflectionCapabilities} from 'angular2/src/reflection/reflection_capabilities';
|
import {ReflectionCapabilities} from 'angular2/src/reflection/reflection_capabilities';
|
||||||
import {httpBindings} from 'angular2/http';
|
import {httpInjectables} from 'angular2/http';
|
||||||
import {Http} from 'angular2/src/http/http';
|
|
||||||
import {IHttp} from 'angular2/src/http/interfaces';
|
|
||||||
import {Response} from 'angular2/src/http/static_response';
|
|
||||||
import {LocalVariable} from './assign_local_directive';
|
|
||||||
import {RxPipeFactory} from './rx_pipe';
|
|
||||||
import {HttpCmp} from './http_comp';
|
import {HttpCmp} from './http_comp';
|
||||||
|
|
||||||
export function main() {
|
export function main() {
|
||||||
reflector.reflectionCapabilities = new ReflectionCapabilities();
|
reflector.reflectionCapabilities = new ReflectionCapabilities();
|
||||||
defaultPipes.rx = [new RxPipeFactory()] bootstrap(
|
bootstrap(HttpCmp, [httpInjectables]);
|
||||||
HttpCmp, [bind(PipeRegistry).toValue(new PipeRegistry(defaultPipes))]);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,15 +1,13 @@
|
||||||
import {HttpCmp} from './http_comp';
|
import {HttpCmp} from './http_comp';
|
||||||
import {bootstrap} from 'angular2/angular2';
|
import {bootstrap} from 'angular2/angular2';
|
||||||
import {bind} from 'angular2/di';
|
|
||||||
import {reflector} from 'angular2/src/reflection/reflection';
|
import {reflector} from 'angular2/src/reflection/reflection';
|
||||||
import {ReflectionCapabilities} from 'angular2/src/reflection/reflection_capabilities';
|
import {ReflectionCapabilities} from 'angular2/src/reflection/reflection_capabilities';
|
||||||
import {PipeRegistry, defaultPipes} from 'angular2/change_detection';
|
import {httpInjectables} from 'angular2/http';
|
||||||
import {RxPipeFactory} from './rx_pipe';
|
|
||||||
|
|
||||||
export function main() {
|
export function main() {
|
||||||
// This entry point is not transformed and exists for testing purposes.
|
// This entry point is not transformed and exists for testing purposes.
|
||||||
// See index.js for an explanation.
|
// See index.js for an explanation.
|
||||||
reflector.reflectionCapabilities = new ReflectionCapabilities();
|
reflector.reflectionCapabilities = new ReflectionCapabilities();
|
||||||
defaultPipes.rx = [new RxPipeFactory()] bootstrap(
|
bootstrap(HttpCmp, [httpInjectables]);
|
||||||
HttpCmp, [bind(PipeRegistry).toValue(new PipeRegistry(defaultPipes))]);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,38 +0,0 @@
|
||||||
/// <reference path="../../../angular2/typings/rx/rx.all.d.ts" />
|
|
||||||
|
|
||||||
import {isBlank, isPresent, CONST} from 'angular2/src/facade/lang';
|
|
||||||
import {Observable, ObservableWrapper} from 'angular2/src/facade/async';
|
|
||||||
import {Pipe, WrappedValue, PipeFactory} from 'angular2/src/change_detection/pipes/pipe';
|
|
||||||
import {ObservablePipe} from 'angular2/src/change_detection/pipes/observable_pipe';
|
|
||||||
import {ChangeDetectorRef} from 'angular2/src/change_detection/change_detector_ref';
|
|
||||||
import * as Rx from 'rx';
|
|
||||||
|
|
||||||
export class RxPipe extends ObservablePipe {
|
|
||||||
supports(obs): boolean {
|
|
||||||
if (Rx.hasOwnProperty('default')) {
|
|
||||||
return obs instanceof (<any>Rx).default.Rx.Observable;
|
|
||||||
} else {
|
|
||||||
return obs instanceof <any>Rx.Observable;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_subscribe(obs): void {
|
|
||||||
this._observable = obs;
|
|
||||||
this._subscription =
|
|
||||||
(<any>obs).subscribe(value => {this._updateLatestValue(value)}, e => { throw e; });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Provides a factory for [ObervablePipe].
|
|
||||||
*
|
|
||||||
* @exportedAs angular2/pipes
|
|
||||||
*/
|
|
||||||
@CONST()
|
|
||||||
export class RxPipeFactory extends PipeFactory {
|
|
||||||
constructor() { super(); }
|
|
||||||
|
|
||||||
supports(obs): boolean { return obs instanceof (<any>Rx).default.Rx.Observable }
|
|
||||||
|
|
||||||
create(cdRef): Pipe { return new RxPipe(cdRef); }
|
|
||||||
}
|
|
Loading…
Reference in New Issue