1 import { Injectable } from '@angular/core';
2 import { Http, Headers, RequestOptionsArgs, Response } from '@angular/http';
3 import { Location } from '@angular/common';
4 import { Observable } from 'rxjs/Observable';
5 import { Subject } from 'rxjs/Subject';
6 import { BehaviorSubject } from 'rxjs/BehaviorSubject';
7 import * as io from 'socket.io-client';
9 import { AlertService } from './alert.service';
10 import { ISdk } from './sdk.service';
13 // Import RxJs required methods
14 import 'rxjs/add/operator/map';
15 import 'rxjs/add/operator/catch';
16 import 'rxjs/add/observable/throw';
17 import 'rxjs/add/operator/mergeMap';
20 export interface IXDSConfigProject {
23 hostSyncThingID: string;
25 defaultSdkID?: string;
28 interface IXDSBuilderConfig {
34 interface IXDSFolderConfig {
40 builderSThgID?: string;
45 interface IXDSConfig {
47 builder: IXDSBuilderConfig;
48 folders: IXDSFolderConfig[];
51 export interface IXDSAgentTarball {
56 export interface IXDSAgentInfo {
57 tarballs: IXDSAgentTarball[];
60 export interface ISdkMessage {
66 export interface ICmdOutput {
73 export interface ICmdExit {
80 export interface IServerStatus {
81 WS_connected: boolean;
85 const FOLDER_TYPE_CLOUDSYNC = 2;
88 export class XDSServerService {
90 public CmdOutput$ = <Subject<ICmdOutput>>new Subject();
91 public CmdExit$ = <Subject<ICmdExit>>new Subject();
92 public Status$: Observable<IServerStatus>;
94 private baseUrl: string;
95 private wsUrl: string;
96 private _status = { WS_connected: false };
97 private statusSubject = <BehaviorSubject<IServerStatus>>new BehaviorSubject(this._status);
100 private socket: SocketIOClient.Socket;
102 constructor(private http: Http, private _window: Window, private alert: AlertService) {
104 this.Status$ = this.statusSubject.asObservable();
106 this.baseUrl = this._window.location.origin + '/api/v1';
107 let re = this._window.location.origin.match(/http[s]?:\/\/([^\/]*)[\/]?/);
108 if (re === null || re.length < 2) {
109 console.error('ERROR: cannot determine Websocket url');
111 this.wsUrl = 'ws://' + re[1];
112 this._handleIoSocket();
116 private _WSState(sts: boolean) {
117 this._status.WS_connected = sts;
118 this.statusSubject.next(Object.assign({}, this._status));
121 private _handleIoSocket() {
122 this.socket = io(this.wsUrl, { transports: ['websocket'] });
124 this.socket.on('connect_error', (res) => {
125 this._WSState(false);
126 console.error('WS Connect_error ', res);
129 this.socket.on('connect', (res) => {
133 this.socket.on('disconnection', (res) => {
134 this._WSState(false);
135 this.alert.error('WS disconnection: ' + res);
138 this.socket.on('error', (err) => {
139 console.error('WS error:', err);
142 this.socket.on('make:output', data => {
143 this.CmdOutput$.next(Object.assign({}, <ICmdOutput>data));
146 this.socket.on('make:exit', data => {
147 this.CmdExit$.next(Object.assign({}, <ICmdExit>data));
150 this.socket.on('exec:output', data => {
151 this.CmdOutput$.next(Object.assign({}, <ICmdOutput>data));
154 this.socket.on('exec:exit', data => {
155 this.CmdExit$.next(Object.assign({}, <ICmdExit>data));
160 getSdks(): Observable<ISdk[]> {
161 return this._get('/sdks');
164 getXdsAgentInfo(): Observable<IXDSAgentInfo> {
165 return this._get('/xdsagent/info');
168 getProjects(): Observable<IXDSFolderConfig[]> {
169 return this._get('/folders');
172 addProject(cfg: IXDSConfigProject): Observable<IXDSFolderConfig> {
173 let folder: IXDSFolderConfig = {
175 label: cfg.label || "",
177 type: FOLDER_TYPE_CLOUDSYNC,
178 syncThingID: cfg.hostSyncThingID,
179 defaultSdkID: cfg.defaultSdkID || "",
181 return this._post('/folder', folder);
184 deleteProject(id: string): Observable<IXDSFolderConfig> {
185 return this._delete('/folder/' + id);
188 exec(prjID: string, dir: string, cmd: string, sdkid?: string, args?: string[], env?: string[]): Observable<any> {
189 return this._post('/exec',
200 make(prjID: string, dir: string, sdkid?: string, args?: string[], env?: string[]): Observable<any> {
201 return this._post('/make',
212 private _attachAuthHeaders(options?: any) {
213 options = options || {};
214 let headers = options.headers || new Headers();
215 // headers.append('Authorization', 'Basic ' + btoa('username:password'));
216 headers.append('Accept', 'application/json');
217 headers.append('Content-Type', 'application/json');
218 // headers.append('Access-Control-Allow-Origin', '*');
220 options.headers = headers;
224 private _get(url: string): Observable<any> {
225 return this.http.get(this.baseUrl + url, this._attachAuthHeaders())
226 .map((res: Response) => res.json())
227 .catch(this._decodeError);
229 private _post(url: string, body: any): Observable<any> {
230 return this.http.post(this.baseUrl + url, JSON.stringify(body), this._attachAuthHeaders())
231 .map((res: Response) => res.json())
233 return this._decodeError(error);
236 private _delete(url: string): Observable<any> {
237 return this.http.delete(this.baseUrl + url, this._attachAuthHeaders())
238 .map((res: Response) => res.json())
239 .catch(this._decodeError);
242 private _decodeError(err: any) {
244 if (typeof err === "object") {
245 if (err.statusText) {
247 } else if (err.error) {
248 e = String(err.error);
250 e = JSON.stringify(err);
253 e = err.json().error || 'Server error';
255 return Observable.throw(e);