'use strict'; /* global angular */ /* global EventSource */ angular.module('Application').service('Client', ['$http', '$interval', 'md5', 'Notification', function ($http, $interval, md5, Notification) { var client = null; // variable available only here to avoid this._property pattern var token = null; // Keep this in sync with docs and constants.js, docker.js var DEFAULT_MEMORY_LIMIT = 1024 * 1024 * 256; function ClientError(statusCode, messageOrObject) { Error.call(this); this.name = this.constructor.name; this.statusCode = statusCode; if (messageOrObject === null || typeof messageOrObject === 'undefined') { this.message = 'Empty message or object'; } else if (typeof messageOrObject === 'string') { this.message = messageOrObject; } else if (messageOrObject.message) { this.message = messageOrObject.message; } else { this.message = JSON.stringify(messageOrObject); } } function defaultErrorHandler(callback) { return function (data, status) { if (status === 401) return client.login(); if (status === 503) { // this could indicate a update/upgrade/restore/migration client.progress(function (error, result) { if (error) { client.error(error); return callback(new ClientError(status, data)); } if (result.update && result.update.percent !== -1) window.location.href = '/update.html'; else callback(new ClientError(status, data)); }, function (data, status) { client.error(data); return callback(new ClientError(status, data)); }); return; } if (status >= 500) { client.error(data); return callback(new ClientError(status, data)); } var obj = data; try { obj = JSON.parse(data); } catch (e) {} callback(new ClientError(status, obj)); }; } // XHR wrapper to set the auth header function get(url, config) { config = config || {}; config.headers = config.headers || {}; config.headers.Authorization = 'Bearer ' + token; return $http.get(client.apiOrigin + url, config); } function head(url, config) { config = config || {}; config.headers = config.headers || {}; config.headers.Authorization = 'Bearer ' + token; return $http.head(client.apiOrigin + url, config); } function post(url, data, config) { data = data || {}; config = config || {}; config.headers = config.headers || {}; config.headers.Authorization = 'Bearer ' + token; return $http.post(client.apiOrigin + url, data, config); } function put(url, data, config) { data = data || {}; config = config || {}; config.headers = config.headers || {}; config.headers.Authorization = 'Bearer ' + token; return $http.put(client.apiOrigin + url, data, config); } function del(url, config) { config = config || {}; config.headers = config.headers || {}; config.headers.Authorization = 'Bearer ' + token; return $http.delete(client.apiOrigin + url, config); } function Client() { this._ready = false; this._configListener = []; this._appsListener = []; this._readyListener = []; this._userInfo = { id: null, username: null, email: null, admin: false }; this._config = { apiServerOrigin: null, webServerOrigin: null, fqdn: null, ip: null, revision: null, update: { box: null, apps: null }, progress: {}, region: null, size: null, memory: 0 }; this._installedApps = []; this._clientId = '<%= oauth.clientId %>'; this._clientSecret = '<%= oauth.clientSecret %>'; // window.location fallback for websocket connections which do not have relative uris this.apiOrigin = '<%= oauth.apiOrigin %>' || window.location.origin; this.avatar = ''; this._refreshConfigTimer = null; this.resetAvatar(); this.setToken(localStorage.token); } Client.prototype.error = function (error) { var message = ''; if (typeof error === 'object') { message = error.message || error; } else { message = error; } Notification.error({ title: 'Cloudron Error', message: message }); }; /* Example usage with an action: var actionScope = $scope.$new(true); actionScope.action = '/#/certs'; Client.notify('title', 'message', true, actionScope); */ Client.prototype.notify = function (title, message, persitent, type, actionScope) { var options = { title: title, message: message}; if (persitent) options.delay = 'never'; // any non Number means never timeout if (actionScope) { if (typeof actionScope.action !== 'string') throw('an actionScope has to have an action url'); options.scope = actionScope; } if (type === 'error') Notification.error(options); else if (type === 'success') Notification.success(options); else if (type === 'info') Notification.info(options); else if (type === 'warning') Notification.warning(options); else throw('Invalid notification type "' + type + '"'); }; Client.prototype.setReady = function () { if (this._ready) return; this._ready = true; this._readyListener.forEach(function (callback) { callback(); }); // clear the listeners, we only callback once! this._readyListener = []; }; Client.prototype.onReady = function (callback) { if (this._ready) callback(); else this._readyListener.push(callback); }; Client.prototype.onConfig = function (callback) { this._configListener.push(callback); if (this._config && this._config.apiServerOrigin) callback(this._config); }; Client.prototype.onApps = function (callback) { this._appsListener.push(callback); callback(this._installedApps); }; Client.prototype.resetAvatar = function () { this.avatar = this.apiOrigin + '/api/v1/cloudron/avatar?' + String(Math.random()).slice(2); var favicon = $('#favicon'); if (favicon) favicon.attr('href', this.avatar); }; Client.prototype.setUserInfo = function (userInfo) { // In order to keep the angular bindings alive, set each property individually this._userInfo.id = userInfo.id; this._userInfo.username = userInfo.username; this._userInfo.email = userInfo.email; this._userInfo.fallbackEmail = userInfo.fallbackEmail; this._userInfo.displayName = userInfo.displayName; this._userInfo.admin = !!userInfo.admin; this._userInfo.gravatar = 'https://www.gravatar.com/avatar/' + md5.createHash(userInfo.email) + '.jpg?s=24&d=mm'; this._userInfo.gravatarHuge = 'https://www.gravatar.com/avatar/' + md5.createHash(userInfo.email) + '.jpg?s=128&d=mm'; }; Client.prototype.setConfig = function (config) { var that = this; // provide fallback to caas if (!config.provider) config.provider = 'caas'; angular.copy(config, this._config); this._configListener.forEach(function (callback) { callback(that._config); }); }; Client.prototype.getInstalledApps = function () { return this._installedApps; }; Client.prototype.getUserInfo = function () { return this._userInfo; }; Client.prototype.getConfig = function () { return this._config; }; Client.prototype.setToken = function (accessToken) { if (!accessToken) localStorage.removeItem('token'); else localStorage.token = accessToken; // set the token closure token = accessToken; }; Client.prototype.getToken = function () { return token; }; /* * Rest API wrappers */ Client.prototype.config = function (callback) { get('/api/v1/cloudron/config').success(function(data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.userInfo = function (callback) { get('/api/v1/profile').success(function(data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.changeCloudronAvatar = function (avatarFile, callback) { var fd = new FormData(); fd.append('avatar', avatarFile); post('/api/v1/settings/cloudron_avatar', fd, { headers: { 'Content-Type': undefined }, transformRequest: angular.identity }).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.changeCloudronName = function (name, callback) { var data = { name: name }; post('/api/v1/settings/cloudron_name', data).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.installApp = function (id, manifest, title, config, callback) { var that = this; var data = { appStoreId: id + '@' + manifest.version, location: config.location, domain: config.domain, portBindings: config.portBindings, accessRestriction: config.accessRestriction, cert: config.cert, key: config.key, sso: config.sso }; post('/api/v1/apps/install', data).success(function (data, status) { if (status !== 202 || typeof data !== 'object') return defaultErrorHandler(callback); // put new app with amended title in cache data.manifest = { title: title }; var icons = that.getAppIconUrls(data); data.iconUrl = icons.cloudron; data.iconUrlStore = icons.store; data.progress = 0; callback(null, data.id); }).error(defaultErrorHandler(callback)); }; Client.prototype.restoreApp = function (appId, backupId, password, callback) { var data = { password: password, backupId: backupId }; post('/api/v1/apps/' + appId + '/restore', data).success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.uninstallApp = function (appId, password, callback) { var data = { password: password }; post('/api/v1/apps/' + appId + '/uninstall', data).success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.configureApp = function (id, config, callback) { var data = { appId: id, location: config.location, domain: config.domain, portBindings: config.portBindings, accessRestriction: config.accessRestriction, cert: config.cert, key: config.key, memoryLimit: config.memoryLimit, altDomain: config.altDomain || null, xFrameOptions: config.xFrameOptions, robotsTxt: config.robotsTxt || null, enableBackup: config.enableBackup }; post('/api/v1/apps/' + id + '/configure', data).success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.updateApp = function (id, manifest, callback) { var data = { appStoreId: manifest.id + '@' + manifest.version }; post('/api/v1/apps/' + id + '/update', data).success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.startApp = function (id, callback) { post('/api/v1/apps/' + id + '/start').success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.stopApp = function (id, callback) { post('/api/v1/apps/' + id + '/stop').success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.debugApp = function (id, state, callback) { var data = { appId: id, debugMode: state ? { readonlyRootfs: true, cmd: [ '/bin/bash', '-c', 'echo "Repair mode. Use the webterminal or cloudron exec to repair. Sleeping" && sleep infinity' ] } : null }; post('/api/v1/apps/' + id + '/configure', data).success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.progress = function (callback, errorCallback) { // this is used in the defaultErrorHandler itself, and avoids a loop if (typeof errorCallback !== 'function') errorCallback = defaultErrorHandler(callback); get('/api/v1/cloudron/progress').success(function(data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(errorCallback); }; Client.prototype.version = function (callback) { get('/api/v1/cloudron/status').success(function(data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.getStatus = function (callback) { get('/api/v1/cloudron/status').success(function(data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.makeURL = function (url) { if (url.indexOf('?') === -1) { return this.apiOrigin + url + '?access_token=' + token; } else { return this.apiOrigin + url + '&access_token=' + token; } }; Client.prototype.setBackupConfig = function (backupConfig, callback) { post('/api/v1/settings/backup_config', backupConfig).success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getBackupConfig = function (callback) { get('/api/v1/settings/backup_config').success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.setAutoupdatePattern = function (pattern, callback) { post('/api/v1/settings/autoupdate_pattern', { pattern: pattern }).success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.checkForUpdates = function (callback) { var that = this; if (that._refreshConfigTimer) $interval.cancel(that._refreshConfigTimer); that._refreshConfigTimer = null; post('/api/v1/cloudron/check_for_updates', { }).success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); that._refreshConfigTimer = $interval(that.refreshConfig.bind(that), 5000, 20 /* 20 times */); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getAutoupdatePattern = function (callback) { get('/api/v1/settings/autoupdate_pattern').success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.setAppstoreConfig = function (config, callback) { var data = config; post('/api/v1/settings/appstore_config', data).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getAppstoreConfig = function (callback) { get('/api/v1/settings/appstore_config').success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.addAuthorizedKey = function (key, callback) { put('/api/v1/cloudron/ssh/authorized_keys', { key: key }).success(function (data, status) { if (status !== 201) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.delAuthorizedKey = function (identifier, callback) { del('/api/v1/cloudron/ssh/authorized_keys/' + identifier).success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getAuthorizedKeys = function (callback) { get('/api/v1/cloudron/ssh/authorized_keys').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.keys); }).error(defaultErrorHandler(callback)); }; Client.prototype.getBackups = function (callback) { get('/api/v1/backups').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.backups); }).error(defaultErrorHandler(callback)); }; Client.prototype.backup = function (callback) { post('/api/v1/backups').success(function(data, status) { if (status !== 202 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.restore = function (backupConfig, backupId, version, callback) { var data = { backupConfig: backupConfig, backupId: backupId, version: version }; post('/api/v1/cloudron/restore', data).success(function(data, status) { if (status !== 200) return callback(new ClientError(status)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getEventLogs = function (action, search, page, perPage, callback) { var config = { params: { action: action, search: search, page: page, per_page: perPage } }; get('/api/v1/cloudron/eventlog', config).success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.eventlogs); }).error(defaultErrorHandler(callback)); }; Client.prototype.getPlatformLogs = function (units, follow, lines, callback) { if (follow) { var eventSource = new EventSource(client.apiOrigin + '/api/v1/cloudron/logstream?lines=' + lines + '&access_token=' + token + '&units=' + units); callback(null, eventSource); } else { get('/api/v1/cloudron/logs?lines=100&units=' + units).success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); } }; Client.prototype.getApps = function (callback) { get('/api/v1/apps').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.apps); }).error(defaultErrorHandler(callback)); }; Client.prototype.getAppLogs = function (appId, follow, lines, callback) { if (follow) { var eventSource = new EventSource(client.apiOrigin + '/api/v1/apps/' + appId + '/logstream?lines=' + lines + '&access_token=' + token); callback(null, eventSource); } else { get('/api/v1/apps/' + appId + '/logs').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); } }; Client.prototype.getAppBackups = function (appId, callback) { get('/api/v1/apps/' + appId + '/backups').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.backups); }).error(defaultErrorHandler(callback)); }; Client.prototype.getUsers = function (callback) { get('/api/v1/users').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.users); }).error(defaultErrorHandler(callback)); }; Client.prototype.getGroups = function (callback) { get('/api/v1/groups').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.groups); }).error(defaultErrorHandler(callback)); }; Client.prototype.setGroups = function (userId, groupIds, callback) { put('/api/v1/users/' + userId + '/groups', { groupIds: groupIds }).success(function (data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getGroup = function (groupId, callback) { get('/api/v1/groups/' + groupId).success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.createGroup = function (name, callback) { var data = { name: name }; post('/api/v1/groups', data).success(function(data, status) { if (status !== 201 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.removeGroup = function (groupId, password, callback) { var config = { data: { password: password }, headers: { 'Content-Type': 'application/json' } }; del('/api/v1/groups/' + groupId, config).success(function(data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.getApp = function (appId, callback) { var appFound = null; this._installedApps.some(function (app) { if (app.id === appId) { appFound = app; return true; } else { return false; } }); if (appFound) return callback(null, appFound); else return callback(new Error('App not found')); }; Client.prototype.getAppIconUrls = function (app) { return { cloudron: app.iconUrl ? (this.apiOrigin + app.iconUrl + '?access_token=' + token) : null, store: app.appStoreId ? (this._config.apiServerOrigin + '/api/v1/apps/' + app.appStoreId + '/versions/' + app.manifest.version + '/icon') : null }; }; Client.prototype.sendInvite = function (user, callback) { post('/api/v1/users/' + user.id + '/invite').success(function (data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data.resetToken); }).error(defaultErrorHandler(callback)); }; Client.prototype.setupDnsConfig = function (domain, zoneName, provider, dnsConfig, callback) { var data = { domain: domain, adminFqdn: 'my.' + domain, zoneName: zoneName, provider: provider, config: dnsConfig }; post('/api/v1/cloudron/dns_setup', data).success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.createAdmin = function (username, password, email, displayName, setupToken, callback) { var that = this; var data = { username: username, password: password, email: email, displayName: displayName }; var query = ''; if (setupToken) query = '?setupToken=' + setupToken; post('/api/v1/cloudron/activate' + query, data).success(function(data, status) { if (status !== 201 || typeof data !== 'object') return callback(new ClientError(status, data)); that.setToken(data.token); that.setUserInfo({ username: username, email: email, admin: true }); callback(null, data.activated); }).error(defaultErrorHandler(callback)); }; Client.prototype.getOAuthClients = function (callback) { get('/api/v1/oauth/clients').success(function(data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.clients); }).error(defaultErrorHandler(callback)); }; Client.prototype.createOAuthClient = function (appId, scope, redirectURI, callback) { var data = { appId: appId, scope: scope, redirectURI: redirectURI }; post('/api/v1/oauth/clients', data).success(function(data, status) { if (status !== 201 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.clients); }).error(defaultErrorHandler(callback)); }; Client.prototype.delOAuthClient = function (id, callback) { del('/api/v1/oauth/clients/' + id).success(function(data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.createTokenByClientId = function (id, expiresAt, callback) { post('/api/v1/oauth/clients/' + id + '/tokens?expiresAt=' + expiresAt).success(function(data, status) { if (status !== 201) return callback(new ClientError(status, data)); callback(null, data.token); }).error(defaultErrorHandler(callback)); }; Client.prototype.getTokensByClientId = function (id, callback) { get('/api/v1/oauth/clients/' + id + '/tokens').success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data.tokens); }).error(defaultErrorHandler(callback)); }; Client.prototype.delTokensByClientId = function (id, callback) { del('/api/v1/oauth/clients/' + id + '/tokens').success(function(data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.delToken = function (clientId, tokenId, callback) { del('/api/v1/oauth/clients/' + clientId + '/tokens/' + tokenId).success(function(data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.update = function (callback) { var data = { }; post('/api/v1/cloudron/update', data).success(function(data, status) { if (status !== 202 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.reboot = function (callback) { post('/api/v1/cloudron/reboot').success(function(data, status) { if (status !== 202 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.changePlan = function (options, password, callback) { var data = options; data.password = password; post('/api/v1/caas/change_plan', data).success(function(data, status) { if (status !== 202 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.setAdmin = function (domain, password, callback) { post('/api/v1/domains/' + domain + '/set_admin', { password: password }).success(function(data, status) { if (status !== 202 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.setCertificate = function (certificateFile, keyFile, callback) { var data = { cert: certificateFile, key: keyFile }; post('/api/v1/settings/certificate', data).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.disks = function (callback) { get('/api/v1/cloudron/disks').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.graphs = function (targets, from, callback) { var config = { params: { target: targets, format: 'json', from: from } }; get('/api/v1/cloudron/graphs', config).success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.feedback = function (type, subject, description, callback) { var data = { type: type, subject: subject, description: description }; post('/api/v1/feedback', data).success(function (data, status) { if (status !== 201) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.createUser = function (username, email, displayName, sendInvite, callback) { var data = { email: email, displayName: displayName, invite: !!sendInvite }; if (username !== null) data.username = username; post('/api/v1/users', data).success(function(data, status) { if (status !== 201 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.updateUser = function (user, callback) { var data = { email: user.email, displayName: user.displayName, fallbackEmail: user.fallbackEmail }; post('/api/v1/users/' + user.id, data).success(function(data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.removeUser = function (userId, password, callback) { var config = { data: { password: password }, headers: { 'Content-Type': 'application/json' } }; del('/api/v1/users/' + userId, config).success(function(data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.updateProfile = function (data, callback) { post('/api/v1/profile', data).success(function(data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.changePassword = function (currentPassword, newPassword, callback) { var data = { password: currentPassword, newPassword: newPassword }; post('/api/v1/profile/password', data).success(function(data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.refreshUserInfo = function (callback) { var that = this; callback = typeof callback === 'function' ? callback : function () {}; this.userInfo(function (error, result) { if (error) return callback(error); that.setUserInfo(result); callback(null); }); }; Client.prototype.refreshConfig = function (callback) { var that = this; callback = typeof callback === 'function' ? callback : function () {}; this.config(function (error, result) { if (error) return callback(error); that.setConfig(result); callback(null); }); }; Client.prototype.appPostProcess = function (app) { // calculate the icon paths var icons = this.getAppIconUrls(app); app.iconUrl = icons.cloudron; app.iconUrlStore = icons.store; // FIXME have a real message structure, not some string to randomly parse // extract progress percentage var installationProgress = app.installationProgress || ''; var progress = parseInt(installationProgress.split(',')[0], 10); // Unfortunatelly some errors are not actual progress messages, but still have a number in fron like a http status code if (isNaN(progress) || progress > 100) { app.progress = 0; app.message = installationProgress; } else { app.progress = progress; app.message = installationProgress.replace(/.*, /,''); } return app; }; Client.prototype.refreshInstalledApps = function (callback) { var that = this; callback = typeof callback === 'function' ? callback : function () {}; this.getApps(function (error, apps) { if (error) return callback(error); // insert or update new apps apps.forEach(function (app) { var found = false; for (var i = 0; i < that._installedApps.length; ++i) { if (that._installedApps[i].id === app.id) { found = i; break; } } var tmp = {}; angular.copy(app, tmp); that.appPostProcess(tmp); // only replace if the app is already known if (found !== false) { angular.copy(tmp, that._installedApps[found]); } else { that._installedApps.push(tmp); } }); // filter out old entries, going backwards to allow splicing for(var i = that._installedApps.length - 1; i >= 0; --i) { if (!apps.some(function (elem) { return (elem.id === that._installedApps[i].id); })) { that._installedApps.splice(i, 1); } } that._installedApps = that._installedApps.sort(function (app1, app2) { return app1.fqdn.localeCompare(app2.fqdn); }); that._appsListener.forEach(function (callback) { callback(that._installedApps); }); callback(null); }); }; Client.prototype.login = function () { this.setToken(null); this._userInfo = {}; var callbackURL = window.location.protocol + '//' + window.location.host + '/login_callback.html'; var scope = 'root,profile,apps'; // generate a state id to protect agains csrf var state = Math.floor((1 + Math.random()) * 0x1000000000000).toString(16).substring(1); window.localStorage.oauth2State = state; // stash for further use in login_callback window.localStorage.returnTo = '/' + window.location.hash; window.location.href = this.apiOrigin + '/api/v1/oauth/dialog/authorize?response_type=token&client_id=' + this._clientId + '&redirect_uri=' + callbackURL + '&scope=' + scope + '&state=' + state; }; Client.prototype.logout = function () { this.setToken(null); this._userInfo = {}; // logout from OAuth session var origin = window.location.protocol + '//' + window.location.host; window.location.href = this.apiOrigin + '/api/v1/session/logout?redirect=' + origin; }; Client.prototype.exchangeCodeForToken = function (code, callback) { var data = { grant_type: 'authorization_code', code: code, redirect_uri: window.location.protocol + '//' + window.location.host, client_id: this._clientId, client_secret: this._clientSecret }; post('/api/v1/oauth/token?response_type=token&client_id=' + this._clientId, data).success(function(data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.access_token); }).error(defaultErrorHandler(callback)); }; Client.prototype.enoughResourcesAvailable = function (app) { var needed = app.manifest.memoryLimit || DEFAULT_MEMORY_LIMIT; // RAM+Swap var used = this.getInstalledApps().reduce(function (prev, cur) { return prev + (cur.memoryLimit || cur.manifest.memoryLimit || DEFAULT_MEMORY_LIMIT); }, 0); var roundedMemory = Math.round(this.getConfig().memory / (1024 * 1024 * 1024)) * 1024 * 1024 * 1024; // round to nearest GB var totalMemory = roundedMemory * 1.2; // cloudron-system-setup.sh creates equal amount of swap. 1.2 factor is arbitrary var available = (totalMemory || 0) - used; return (available - needed) >= 0; }; Client.prototype.uploadFile = function (appId, file, progressCallback, callback) { var fd = new FormData(); fd.append('file', file); post('/api/v1/apps/' + appId + '/upload?file=/tmp/' + file.name, fd, { headers: { 'Content-Type': undefined }, transformRequest: angular.identity, uploadEventHandlers: { progress: progressCallback } }).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.checkDownloadableFile = function (appId, filePath, callback) { head('/api/v1/apps/' + appId + '/download?file=' + filePath, { headers: { 'Content-Type': undefined } }).success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.sendTestMail = function (domain, to, callback) { var data = { to: to }; post('/api/v1/cloudron/' + domain + '/send_test_mail', data).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; // Domains Client.prototype.getDomains = function (callback) { get('/api/v1/domains').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data.domains); }).error(defaultErrorHandler(callback)); }; Client.prototype.getDomain = function (domain, callback) { get('/api/v1/domains/' + domain).success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.addMailDomain = function (domain, callback) { post('/api/v1/mail', { domain: domain }).success(function (data, status) { if (status !== 201) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); // this doesn't call defaultErrorHandler till we fix domains code to use this directly }; Client.prototype.addDomain = function (domain, provider, config, fallbackCertificate, callback) { var data = { domain: domain, provider: provider, config: config }; var that = this; if (fallbackCertificate) data.fallbackCertificate = fallbackCertificate; // hack until we fix the domains.js post('/api/v1/domains', data).success(function (data, status) { if (status !== 201 || typeof data !== 'object') return callback(new ClientError(status, data)); that.addMailDomain(domain, callback); }).error(defaultErrorHandler(callback)); }; Client.prototype.updateDomain = function (domain, provider, config, fallbackCertificate, callback) { var data = { provider: provider, config: config }; if (fallbackCertificate) data.fallbackCertificate = fallbackCertificate; put('/api/v1/domains/' + domain, data).success(function (data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.removeMailDomain = function (domain, password, callback) { var config = { data: { password: password }, headers: { 'Content-Type': 'application/json' } }; del('/api/v1/mail/' + domain, config).success(function (data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(callback); // this doesn't call defaultErrorHandler since we want to handle 404 }; Client.prototype.removeDomain = function (domain, password, callback) { var config = { data: { password: password }, headers: { 'Content-Type': 'application/json' } }; this.removeMailDomain(domain, password, function () { // hack: ignore errors until we fix the domains.js del('/api/v1/domains/' + domain, config).success(function (data, status) { if (status !== 204) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }); }; // Email Client.prototype.getMailConfigForDomain = function (domain, callback) { get('/api/v1/mail/' + domain).success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.enableMailForDomain = function (domain, enabled, callback) { post('/api/v1/mail/' + domain + '/enable', { enabled: enabled }).success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getMailStatusForDomain = function (domain, callback) { get('/api/v1/mail/' + domain + '/status').success(function (data, status) { if (status !== 200 || typeof data !== 'object') return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.setMailRelay = function (domain, config, callback) { post('/api/v1/mail/' + domain + '/relay', config).success(function (data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.setCatchallAddresses = function (domain, addresses, callback) { post('/api/v1/mail/' + domain + '/catch_all', { address: addresses }).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; // Mailboxes Client.prototype.getMailboxes = function (domain, callback) { get('/api/v1/mail/' + domain + '/mailboxes').success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data); }).error(defaultErrorHandler(callback)); }; Client.prototype.getUserMailbox = function (domain, userId, callback) { get('/api/v1/mail/' + domain + '/mailboxes/' + userId).success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data.mailbox); }).error(defaultErrorHandler(callback)); }; Client.prototype.enableUserMailbox = function (domain, userId, callback) { post('/api/v1/mail/' + domain + '/mailboxes/' + userId, {}).success(function(data, status) { if (status !== 201) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.disableUserMailbox = function (domain, userId, callback) { del('/api/v1/mail/' + domain + '/mailboxes/' + userId).success(function(data, status) { if (status !== 201) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getAliases = function (domain, userId, callback) { get('/api/v1/mail/' + domain + '/aliases/' + userId).success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data.aliases); }).error(defaultErrorHandler(callback)); }; Client.prototype.setAliases = function (domain, userId, aliases, callback) { var data = { aliases: aliases }; put('/api/v1/mail/' + domain + '/aliases/' + userId, data).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.getMailingList = function (domain, groupId, callback) { get('/api/v1/mail/' + domain + '/lists/' + groupId).success(function(data, status) { if (status !== 200) return callback(new ClientError(status, data)); callback(null, data.list); }).error(defaultErrorHandler(callback)); }; Client.prototype.addMailingList = function (domain, groupId, callback) { var data = { groupId: groupId }; post('/api/v1/mail/' + domain + '/lists', data).success(function(data, status) { if (status !== 201) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; Client.prototype.removeMailingList = function (domain, groupId, callback) { del('/api/v1/mail/' + domain + '/lists/' + groupId).success(function(data, status) { if (status !== 202) return callback(new ClientError(status, data)); callback(null); }).error(defaultErrorHandler(callback)); }; client = new Client(); return client; }]);