2015-07-14 19:53:04 -05:00
|
|
|
import {ConnectionBackend, Connection} from '../interfaces';
|
2015-10-29 17:50:12 -07:00
|
|
|
import {ReadyStates, RequestMethods, ResponseTypes} from '../enums';
|
2015-07-14 19:53:04 -05:00
|
|
|
import {Request} from '../static_request';
|
|
|
|
import {Response} from '../static_response';
|
|
|
|
import {ResponseOptions, BaseResponseOptions} from '../base_response_options';
|
2015-10-10 14:58:46 +02:00
|
|
|
import {Injectable} from 'angular2/angular2';
|
2015-07-14 19:53:04 -05:00
|
|
|
import {BrowserJsonp} from './browser_jsonp';
|
2015-11-06 17:34:07 -08:00
|
|
|
import {makeTypeError} from 'angular2/src/facade/exceptions';
|
|
|
|
import {StringWrapper, isPresent} from 'angular2/src/facade/lang';
|
2015-10-30 23:52:37 -07:00
|
|
|
import {Observable} from 'angular2/angular2';
|
2015-10-29 17:50:12 -07:00
|
|
|
|
|
|
|
const JSONP_ERR_NO_CALLBACK = 'JSONP injected script did not invoke callback.';
|
|
|
|
const JSONP_ERR_WRONG_METHOD = 'JSONP requests must use GET request method.';
|
|
|
|
|
2015-10-06 06:53:39 -07:00
|
|
|
export abstract class JSONPConnection implements Connection {
|
2015-07-14 19:53:04 -05:00
|
|
|
readyState: ReadyStates;
|
|
|
|
request: Request;
|
2015-10-30 23:52:37 -07:00
|
|
|
response: Observable<Response>;
|
2015-10-06 06:53:39 -07:00
|
|
|
abstract finished(data?: any): void;
|
|
|
|
}
|
|
|
|
|
|
|
|
export class JSONPConnection_ extends JSONPConnection {
|
2015-07-14 19:53:04 -05:00
|
|
|
private _id: string;
|
|
|
|
private _script: Element;
|
|
|
|
private _responseData: any;
|
|
|
|
private _finished: boolean = false;
|
|
|
|
|
|
|
|
constructor(req: Request, private _dom: BrowserJsonp,
|
|
|
|
private baseResponseOptions?: ResponseOptions) {
|
2015-10-06 06:53:39 -07:00
|
|
|
super();
|
2015-08-26 13:40:12 -07:00
|
|
|
if (req.method !== RequestMethods.Get) {
|
2015-10-29 17:50:12 -07:00
|
|
|
throw makeTypeError(JSONP_ERR_WRONG_METHOD);
|
2015-07-14 19:53:04 -05:00
|
|
|
}
|
|
|
|
this.request = req;
|
2015-09-25 18:53:32 -04:00
|
|
|
this.response = new Observable(responseObserver => {
|
2015-07-14 19:53:04 -05:00
|
|
|
|
2015-09-25 18:53:32 -04:00
|
|
|
this.readyState = ReadyStates.Loading;
|
|
|
|
let id = this._id = _dom.nextRequestID();
|
2015-07-14 19:53:04 -05:00
|
|
|
|
2015-09-25 18:53:32 -04:00
|
|
|
_dom.exposeConnection(id, this);
|
2015-07-14 19:53:04 -05:00
|
|
|
|
2015-09-25 18:53:32 -04:00
|
|
|
// Workaround Dart
|
|
|
|
// url = url.replace(/=JSONP_CALLBACK(&|$)/, `generated method`);
|
|
|
|
let callback = _dom.requestCallback(this._id);
|
|
|
|
let url: string = req.url;
|
|
|
|
if (url.indexOf('=JSONP_CALLBACK&') > -1) {
|
|
|
|
url = StringWrapper.replace(url, '=JSONP_CALLBACK&', `=${callback}&`);
|
|
|
|
} else if (url.lastIndexOf('=JSONP_CALLBACK') === url.length - '=JSONP_CALLBACK'.length) {
|
2015-10-31 13:04:26 -07:00
|
|
|
url = url.substring(0, url.length - '=JSONP_CALLBACK'.length) + `=${callback}`;
|
2015-07-14 19:53:04 -05:00
|
|
|
}
|
|
|
|
|
2015-09-25 18:53:32 -04:00
|
|
|
let script = this._script = _dom.build(url);
|
2015-07-14 19:53:04 -05:00
|
|
|
|
2015-09-25 18:53:32 -04:00
|
|
|
let onLoad = event => {
|
|
|
|
if (this.readyState === ReadyStates.Cancelled) return;
|
|
|
|
this.readyState = ReadyStates.Done;
|
|
|
|
_dom.cleanup(script);
|
|
|
|
if (!this._finished) {
|
2015-10-29 17:50:12 -07:00
|
|
|
let responseOptions =
|
2015-11-19 18:47:29 -08:00
|
|
|
new ResponseOptions({body: JSONP_ERR_NO_CALLBACK, type: ResponseTypes.Error, url});
|
2015-10-29 17:50:12 -07:00
|
|
|
if (isPresent(baseResponseOptions)) {
|
|
|
|
responseOptions = baseResponseOptions.merge(responseOptions);
|
|
|
|
}
|
|
|
|
responseObserver.error(new Response(responseOptions));
|
2015-09-25 18:53:32 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-11-19 18:47:29 -08:00
|
|
|
let responseOptions = new ResponseOptions({body: this._responseData, url});
|
2015-09-25 18:53:32 -04:00
|
|
|
if (isPresent(this.baseResponseOptions)) {
|
|
|
|
responseOptions = this.baseResponseOptions.merge(responseOptions);
|
|
|
|
}
|
|
|
|
|
|
|
|
responseObserver.next(new Response(responseOptions));
|
|
|
|
responseObserver.complete();
|
|
|
|
};
|
|
|
|
|
|
|
|
let onError = error => {
|
|
|
|
if (this.readyState === ReadyStates.Cancelled) return;
|
|
|
|
this.readyState = ReadyStates.Done;
|
|
|
|
_dom.cleanup(script);
|
2015-10-29 17:50:12 -07:00
|
|
|
let responseOptions = new ResponseOptions({body: error.message, type: ResponseTypes.Error});
|
|
|
|
if (isPresent(baseResponseOptions)) {
|
|
|
|
responseOptions = baseResponseOptions.merge(responseOptions);
|
|
|
|
}
|
|
|
|
responseObserver.error(new Response(responseOptions));
|
2015-09-25 18:53:32 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
script.addEventListener('load', onLoad);
|
|
|
|
script.addEventListener('error', onError);
|
2015-07-14 19:53:04 -05:00
|
|
|
|
2015-09-25 18:53:32 -04:00
|
|
|
_dom.send(script);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
this.readyState = ReadyStates.Cancelled;
|
|
|
|
script.removeEventListener('load', onLoad);
|
|
|
|
script.removeEventListener('error', onError);
|
|
|
|
if (isPresent(script)) {
|
|
|
|
this._dom.cleanup(script);
|
|
|
|
}
|
|
|
|
|
2015-10-03 10:04:17 -07:00
|
|
|
};
|
2015-09-25 18:53:32 -04:00
|
|
|
});
|
2015-07-14 19:53:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
finished(data?: any) {
|
|
|
|
// Don't leak connections
|
|
|
|
this._finished = true;
|
|
|
|
this._dom.removeConnection(this._id);
|
2015-08-26 13:40:12 -07:00
|
|
|
if (this.readyState === ReadyStates.Cancelled) return;
|
2015-07-14 19:53:04 -05:00
|
|
|
this._responseData = data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-06 06:53:39 -07:00
|
|
|
export abstract class JSONPBackend extends ConnectionBackend {}
|
|
|
|
|
2015-07-14 19:53:04 -05:00
|
|
|
@Injectable()
|
2015-10-06 06:53:39 -07:00
|
|
|
export class JSONPBackend_ extends JSONPBackend {
|
|
|
|
constructor(private _browserJSONP: BrowserJsonp, private _baseResponseOptions: ResponseOptions) {
|
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
2015-07-14 19:53:04 -05:00
|
|
|
createConnection(request: Request): JSONPConnection {
|
2015-10-06 06:53:39 -07:00
|
|
|
return new JSONPConnection_(request, this._browserJSONP, this._baseResponseOptions);
|
2015-07-14 19:53:04 -05:00
|
|
|
}
|
|
|
|
}
|