wizard.js 43 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360
  1. /**
  2. * Licensed to the Apache Software Foundation (ASF) under one
  3. * or more contributor license agreements. See the NOTICE file
  4. * distributed with this work for additional information
  5. * regarding copyright ownership. The ASF licenses this file
  6. * to you under the Apache License, Version 2.0 (the
  7. * "License"); you may not use this file except in compliance
  8. * with the License. You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. */
  18. var App = require('app');
  19. require('models/host');
  20. App.WizardController = Em.Controller.extend(App.LocalStorage, App.ThemesMappingMixin, {
  21. isStepDisabled: null,
  22. previousStep: 0,
  23. /**
  24. * map of actions which load data required by which step
  25. * used by <code>loadAllPriorSteps</code>
  26. */
  27. loadMap: {},
  28. /**
  29. * Wizard properties in local storage, which should be cleaned right after wizard has been finished
  30. */
  31. dbPropertiesToClean: [
  32. 'service',
  33. 'hosts',
  34. 'masterComponentHosts',
  35. 'slaveComponentHosts',
  36. 'cluster',
  37. 'allHostNames',
  38. 'installOptions',
  39. 'allHostNamesPattern',
  40. 'serviceComponents',
  41. 'fileNamesToUpdate'
  42. ],
  43. sensibleConfigs: [
  44. { name: 'admin_principal', filename: 'krb5-conf.xml'},
  45. { name: 'admin_password', filename: 'krb5-conf.xml' }
  46. ],
  47. init: function () {
  48. this.clusters = App.Cluster.find();
  49. this.setIsStepDisabled();
  50. },
  51. connectOutlet:function(name, context) {
  52. if (name !== 'loading') this.set('isStepDisabled.isLocked', false);
  53. App.get('router').set('transitionInProgress', false);
  54. App.get('router').set('nextBtnClickInProgress', false);
  55. return this._super.apply(this,arguments);
  56. },
  57. /**
  58. * Set <code>isStepDisabled</code> with list of available steps (basing on <code>totalSteps</code>)
  59. * @method setIsStepDisabled
  60. */
  61. setIsStepDisabled: function () {
  62. this.set('isStepDisabled', Ember.ArrayProxy.create({
  63. content:[],
  64. isLocked:true,
  65. objectAtContent: function(idx) {
  66. var obj = this.get('content').objectAt(idx);
  67. if (obj && !obj.hasOwnProperty('isLocked')) {
  68. obj.reopen({
  69. isLocked:true,
  70. get:function (key) {
  71. return (key === 'value' && this.get('isLocked')) || this._super.apply(this,arguments);
  72. },
  73. notifyValues:function () {
  74. this.notifyPropertyChange('value');
  75. }.observes('isLocked')
  76. });
  77. }
  78. return obj;
  79. },
  80. toggleLock:function () {
  81. this.setEach('isLocked',this.get('isLocked'));
  82. }.observes('isLocked')
  83. }));
  84. this.get('isStepDisabled').pushObject(Em.Object.create({
  85. step: 1,
  86. value: false
  87. }));
  88. for (var i = 2; i <= this.get('totalSteps'); i++) {
  89. this.get('isStepDisabled').pushObject(Em.Object.create({
  90. step: i,
  91. value: true
  92. }));
  93. }
  94. },
  95. slaveComponents: function () {
  96. return App.StackServiceComponent.find().filterProperty('isSlave', true);
  97. }.property('App.router.clusterController.isLoaded'),
  98. allHosts: function () {
  99. var dbHosts = this.get('content.hosts');
  100. var hosts = [];
  101. var hostComponents = [];
  102. for (var hostName in dbHosts) {
  103. hostComponents = [];
  104. var disksOverallCapacity = 0;
  105. var diskFree = 0;
  106. dbHosts[hostName].hostComponents.forEach(function (componentName) {
  107. hostComponents.push(Em.Object.create({
  108. componentName: componentName,
  109. displayName: App.format.role(componentName)
  110. }));
  111. });
  112. dbHosts[hostName].disk_info.forEach(function (disk) {
  113. disksOverallCapacity += parseFloat(disk.size);
  114. diskFree += parseFloat(disk.available);
  115. });
  116. hosts.push(Em.Object.create({
  117. id: hostName,
  118. hostName: hostName,
  119. publicHostName: hostName,
  120. diskInfo: dbHosts[hostName].disk_info,
  121. diskTotal: disksOverallCapacity / (1024 * 1024),
  122. diskFree: diskFree / (1024 * 1024),
  123. disksMounted: dbHosts[hostName].disk_info.length,
  124. cpu: dbHosts[hostName].cpu,
  125. memory: dbHosts[hostName].memory,
  126. osType: dbHosts[hostName].osType ? dbHosts[hostName].osType: 0,
  127. osArch: dbHosts[hostName].osArch ? dbHosts[hostName].osArch : 0,
  128. ip: dbHosts[hostName].ip ? dbHosts[hostName].ip: 0,
  129. hostComponents: hostComponents
  130. }))
  131. }
  132. return hosts;
  133. }.property('content.hosts'),
  134. setStepsEnable: function () {
  135. for (var i = 1; i <= this.totalSteps; i++) {
  136. var step = this.get('isStepDisabled').findProperty('step', i);
  137. if (i <= this.get('currentStep')) {
  138. step.set('value', false);
  139. } else {
  140. step.set('value', true);
  141. }
  142. }
  143. }.observes('currentStep'),
  144. /**
  145. * Enable step link in left nav menu
  146. * @param step - step number
  147. */
  148. enableStep: function (step) {
  149. this.get('isStepDisabled').findProperty('step', step).set('value', false);
  150. },
  151. setLowerStepsDisable: function (stepNo) {
  152. for (var i = 1; i < stepNo; i++) {
  153. var step = this.get('isStepDisabled').findProperty('step', i);
  154. step.set('value', true);
  155. }
  156. },
  157. /**
  158. * Set current step to new value.
  159. * Method moved from App.router.setInstallerCurrentStep
  160. * @param currentStep
  161. * @param completed
  162. */
  163. currentStep: function () {
  164. return App.get('router').getWizardCurrentStep(this.get('name').substr(0, this.get('name').length - 10));
  165. }.property(),
  166. /**
  167. * Set current step to new value.
  168. * Method moved from App.router.setInstallerCurrentStep
  169. * @param currentStep
  170. * @param completed
  171. */
  172. setCurrentStep: function (currentStep, completed) {
  173. this.set('previousStep', this.get('currentStep'));
  174. App.db.setWizardCurrentStep(this.get('name').substr(0, this.get('name').length - 10), currentStep, completed);
  175. this.set('currentStep', currentStep);
  176. },
  177. clusters: null,
  178. isStep0: function () {
  179. return this.get('currentStep') == 0;
  180. }.property('currentStep'),
  181. isStep1: function () {
  182. return this.get('currentStep') == 1;
  183. }.property('currentStep'),
  184. isStep2: function () {
  185. return this.get('currentStep') == 2;
  186. }.property('currentStep'),
  187. isStep3: function () {
  188. return this.get('currentStep') == 3;
  189. }.property('currentStep'),
  190. isStep4: function () {
  191. return this.get('currentStep') == 4;
  192. }.property('currentStep'),
  193. isStep5: function () {
  194. return this.get('currentStep') == 5;
  195. }.property('currentStep'),
  196. isStep6: function () {
  197. return this.get('currentStep') == 6;
  198. }.property('currentStep'),
  199. isStep7: function () {
  200. return this.get('currentStep') == 7;
  201. }.property('currentStep'),
  202. isStep8: function () {
  203. return this.get('currentStep') == 8;
  204. }.property('currentStep'),
  205. isStep9: function () {
  206. return this.get('currentStep') == 9;
  207. }.property('currentStep'),
  208. isStep10: function () {
  209. return this.get('currentStep') == 10;
  210. }.property('currentStep'),
  211. /**
  212. * Move user to the selected step
  213. *
  214. * @param {number} step number of the step, where user is moved
  215. * @param {boolean} disableNaviWarning true - don't show warning about moving more than 1 step back
  216. * @returns {boolean}
  217. */
  218. gotoStep: function (step, disableNaviWarning) {
  219. if (this.get('isStepDisabled').findProperty('step', step).get('value') !== false) {
  220. return false;
  221. }
  222. // if going back from Step 9 in Install Wizard, delete the checkpoint so that the user is not redirected
  223. // to Step 9
  224. if (this.get('content.controllerName') == 'installerController' && this.get('currentStep') === '9' && step < 9) {
  225. App.clusterStatus.setClusterStatus({
  226. clusterName: this.get('clusterName'),
  227. clusterState: 'CLUSTER_NOT_CREATED_1',
  228. wizardControllerName: 'installerController',
  229. localdb: {}
  230. });
  231. }
  232. if ((this.get('currentStep') - step) > 1 && !disableNaviWarning) {
  233. App.ModalPopup.show({
  234. header: Em.I18n.t('installer.navigation.warning.header'),
  235. onPrimary: function () {
  236. App.router.send('gotoStep' + step);
  237. this.hide();
  238. },
  239. body: "If you proceed to go back to Step " + step + ", you will lose any changes you have made beyond this step"
  240. });
  241. } else {
  242. App.router.send('gotoStep' + step);
  243. }
  244. return true;
  245. },
  246. gotoStep0: function () {
  247. this.gotoStep(0);
  248. },
  249. gotoStep1: function () {
  250. this.gotoStep(1);
  251. },
  252. gotoStep2: function () {
  253. this.gotoStep(2);
  254. },
  255. gotoStep3: function () {
  256. this.gotoStep(3);
  257. },
  258. gotoStep4: function () {
  259. this.gotoStep(4);
  260. },
  261. gotoStep5: function () {
  262. this.gotoStep(5);
  263. },
  264. gotoStep6: function () {
  265. this.gotoStep(6);
  266. },
  267. gotoStep7: function () {
  268. this.gotoStep(7);
  269. },
  270. gotoStep8: function () {
  271. this.gotoStep(8);
  272. },
  273. gotoStep9: function () {
  274. this.gotoStep(9);
  275. },
  276. gotoStep10: function () {
  277. this.gotoStep(10);
  278. },
  279. /**
  280. * Initialize host status info for step9
  281. */
  282. setInfoForStep9: function () {
  283. var hostInfo = this.getDBProperty('hosts');
  284. for (var index in hostInfo) {
  285. hostInfo[index].status = "pending";
  286. hostInfo[index].message = 'Waiting';
  287. hostInfo[index].logTasks = [];
  288. hostInfo[index].tasks = [];
  289. hostInfo[index].progress = '0';
  290. }
  291. this.setDBProperty('hosts', hostInfo);
  292. },
  293. /**
  294. * Remove all data for installOptions step
  295. */
  296. clearInstallOptions: function () {
  297. var installOptions = this.getInstallOptions();
  298. this.set('content.installOptions', installOptions);
  299. this.set('content.hosts', {});
  300. this.setDBProperties({
  301. installOptions: installOptions,
  302. hosts: {}
  303. });
  304. },
  305. toObject: function (object) {
  306. var result = {};
  307. for (var i in object) {
  308. if (object.hasOwnProperty(i)) {
  309. result[i] = object[i];
  310. }
  311. }
  312. return result;
  313. },
  314. /**
  315. * Convert any object or array to pure JS instance without inherit properties
  316. * It is used to convert Ember.Object to pure JS Object and Ember.Array to pure JS Array
  317. * @param originalInstance
  318. * @returns {*}
  319. */
  320. toJSInstance: function (originalInstance) {
  321. var convertedInstance = originalInstance;
  322. if (Em.isArray(originalInstance)) {
  323. convertedInstance = [];
  324. originalInstance.forEach(function (element) {
  325. convertedInstance.push(this.toJSInstance(element));
  326. }, this)
  327. } else if (originalInstance && typeof originalInstance === 'object') {
  328. convertedInstance = {};
  329. for (var property in originalInstance) {
  330. if (originalInstance.hasOwnProperty(property)) {
  331. convertedInstance[property] = this.toJSInstance(originalInstance[property]);
  332. }
  333. }
  334. }
  335. return convertedInstance
  336. },
  337. /**
  338. * save status of the cluster. This is called from step8 and step9 to persist install and start requestId
  339. * @param clusterStatus object with status, isCompleted, requestId, isInstallError and isStartError field.
  340. */
  341. saveClusterStatus: function (clusterStatus) {
  342. var oldStatus = this.toObject(this.get('content.cluster'));
  343. clusterStatus = jQuery.extend(oldStatus, clusterStatus);
  344. if (clusterStatus.requestId &&
  345. clusterStatus.oldRequestsId.indexOf(clusterStatus.requestId) === -1) {
  346. clusterStatus.oldRequestsId.push(clusterStatus.requestId);
  347. }
  348. this.set('content.cluster', clusterStatus);
  349. this.setDBProperty('cluster', clusterStatus);
  350. },
  351. /**
  352. * Invoke installation of selected services to the server and saves the request id returned by the server.
  353. * @param isRetry
  354. */
  355. installServices: function (isRetry, callback) {
  356. // clear requests since we are installing services
  357. // and we don't want to get tasks for previous install attempts
  358. this.set('content.cluster.oldRequestsId', []);
  359. var data;
  360. callback = callback || Em.K;
  361. if (isRetry) {
  362. data = {
  363. context: Em.I18n.t('requestInfo.installComponents'),
  364. HostRoles: {"state": "INSTALLED"},
  365. urlParams: "HostRoles/desired_state=INSTALLED"
  366. };
  367. } else {
  368. data = {
  369. context: Em.I18n.t('requestInfo.installServices'),
  370. ServiceInfo: {"state": "INSTALLED"},
  371. urlParams: "ServiceInfo/state=INIT"
  372. };
  373. }
  374. var clusterStatus = {
  375. status: 'PENDING'
  376. };
  377. this.saveClusterStatus(clusterStatus);
  378. App.ajax.send({
  379. name: isRetry ? 'common.host_components.update' : 'common.services.update',
  380. sender: this,
  381. data: data,
  382. success: 'installServicesSuccessCallback',
  383. error: 'installServicesErrorCallback'
  384. }).then(callback, callback);
  385. },
  386. installServicesSuccessCallback: function (jsonData) {
  387. var installStartTime = App.dateTime();
  388. if (jsonData) {
  389. var requestId = jsonData.Requests.id;
  390. var clusterStatus = {
  391. status: 'PENDING',
  392. requestId: requestId,
  393. isInstallError: false,
  394. isCompleted: false,
  395. installStartTime: installStartTime
  396. };
  397. this.saveClusterStatus(clusterStatus);
  398. }
  399. },
  400. installServicesErrorCallback: function (request, ajaxOptions, error) {
  401. var clusterStatus = {
  402. status: 'PENDING',
  403. requestId: this.get('content.cluster.requestId'),
  404. isInstallError: true,
  405. isCompleted: false
  406. };
  407. this.saveClusterStatus(clusterStatus);
  408. App.showAlertPopup(Em.I18n.t('common.errorPopup.header'), request.responseText);
  409. },
  410. /**
  411. * show popup, that display status of bootstrap launching
  412. * @param callback
  413. * @return {Object}
  414. */
  415. showLaunchBootstrapPopup: function (callback) {
  416. return App.ModalPopup.show({
  417. header: Em.I18n.t('installer.step2.bootStrap.header'),
  418. isError: false,
  419. serverError: null,
  420. bodyClass: Em.View.extend({
  421. templateName: require('templates/wizard/bootstrap_call_popup')
  422. }),
  423. showFooter: false,
  424. showCloseButton: false,
  425. secondary: null,
  426. /**
  427. * handle requestId when call is completed,
  428. * if it's correct call callback and hide popup
  429. * otherwise notify error and enable buttons to close popup
  430. * @param requestId
  431. * @param serverError
  432. * @param status
  433. * @param log
  434. */
  435. finishLoading: function (requestId, serverError, status, log) {
  436. if (Em.isNone(requestId) || status == 'ERROR') {
  437. var stepController = App.get('router.wizardStep3Controller');
  438. this.setProperties({
  439. isError: true,
  440. showFooter: true,
  441. showCloseButton: true,
  442. serverError: status == 'ERROR' ? log : serverError
  443. });
  444. stepController.setProperties({
  445. isRegistrationInProgress: false,
  446. isBootstrapFailed: true
  447. });
  448. stepController.get('hosts').setEach('bootStatus', 'FAILED');
  449. } else {
  450. callback(requestId);
  451. this.hide();
  452. }
  453. }
  454. });
  455. },
  456. /**
  457. * Bootstrap selected hosts.
  458. * @param bootStrapData
  459. * @param callback
  460. * @return {Object}
  461. */
  462. launchBootstrap: function (bootStrapData, callback) {
  463. var popup = this.showLaunchBootstrapPopup(callback);
  464. App.ajax.send({
  465. name: 'wizard.launch_bootstrap',
  466. sender: this,
  467. data: {
  468. bootStrapData: bootStrapData,
  469. popup: popup
  470. },
  471. success: 'launchBootstrapSuccessCallback',
  472. error: 'launchBootstrapErrorCallback'
  473. });
  474. return popup;
  475. },
  476. launchBootstrapSuccessCallback: function (data, opt, params) {
  477. params.popup.finishLoading(data.requestId, null, data.status, data.log);
  478. },
  479. launchBootstrapErrorCallback: function (request, ajaxOptions, error, opt, params) {
  480. params.popup.finishLoading(null, error);
  481. },
  482. /**
  483. * Load <code>content.<name></code> variable from localStorage, if wasn't loaded before.
  484. * If you specify <code>reload</code> to true - it will reload it.
  485. * @param name
  486. * @param reload
  487. * @return {Boolean}
  488. */
  489. load: function (name, reload) {
  490. if (this.get('content.' + name) && !reload) {
  491. return false;
  492. }
  493. var result = this.getDBProperty(name);
  494. if (!result) {
  495. if (this['get' + name.capitalize()]) {
  496. result = this['get' + name.capitalize()]();
  497. this.setDBProperty(name, result);
  498. }
  499. else {
  500. console.debug('get' + name.capitalize(), ' not defined in the ' + this.get('name'));
  501. }
  502. }
  503. this.set('content.' + name, result);
  504. },
  505. save: function (name) {
  506. var convertedValue = this.toJSInstance(this.get('content.' + name));
  507. this.setDBProperty(name, convertedValue);
  508. },
  509. clear: function () {
  510. this.set('content', Ember.Object.create({
  511. 'controllerName': this.get('content.controllerName')
  512. }));
  513. this.set('currentStep', 0);
  514. this.clearStorageData();
  515. },
  516. clusterStatusTemplate: {
  517. name: "",
  518. status: "PENDING",
  519. isCompleted: false,
  520. requestId: null,
  521. installStartTime: null,
  522. installTime: null,
  523. isInstallError: false,
  524. isStartError: false,
  525. oldRequestsId: []
  526. },
  527. clearStorageData: function () {
  528. var hash = {};
  529. this.get('dbPropertiesToClean').forEach(function (key) {
  530. hash[key] = undefined;
  531. }, this);
  532. this.setDBProperties(hash);
  533. },
  534. getInstallOptions: function() {
  535. return jQuery.extend({}, App.get('isHadoopWindowsStack') ? this.get('installWindowsOptionsTemplate') : this.get('installOptionsTemplate'));
  536. },
  537. installOptionsTemplate: {
  538. hostNames: "", //string
  539. manualInstall: false, //true, false
  540. useSsh: true, //bool
  541. javaHome: App.defaultJavaHome, //string
  542. localRepo: false, //true, false
  543. sshKey: "", //string
  544. bootRequestId: null, //string
  545. sshUser: "root", //string
  546. sshPort: "22",
  547. agentUser: "root" //string
  548. },
  549. installWindowsOptionsTemplate: {
  550. hostNames: "", //string
  551. manualInstall: false, //true, false
  552. useSsh: true, //bool
  553. javaHome: App.defaultJavaHome, //string
  554. localRepo: false, //true, false
  555. sshKey: "", //string
  556. bootRequestId: null, //string
  557. sshUser: "", //string
  558. sshPort: "22",
  559. agentUser: "" //string
  560. },
  561. loadedServiceComponents: null,
  562. /**
  563. * Generate serviceComponents as pr the stack definition and save it to localdata
  564. * called form stepController step4WizardController
  565. */
  566. loadServiceComponents: function () {
  567. return App.ajax.send({
  568. name: 'wizard.service_components',
  569. sender: this,
  570. data: {
  571. stackUrl: App.get('stackVersionURL'),
  572. stackVersion: App.get('currentStackVersionNumber')
  573. },
  574. success: 'loadServiceComponentsSuccessCallback',
  575. error: 'loadServiceComponentsErrorCallback'
  576. });
  577. },
  578. loadServiceComponentsSuccessCallback: function (jsonData) {
  579. var props = this.getDBProperties(['selectedServiceNames', 'installedServiceNames']);
  580. var savedSelectedServices = props.selectedServiceNames;
  581. var savedInstalledServices = props.installedServiceNames;
  582. this.set('content.selectedServiceNames', savedSelectedServices);
  583. this.set('content.installedServiceNames', savedInstalledServices);
  584. if (!savedSelectedServices) {
  585. jsonData.items.forEach(function (service) {
  586. service.StackServices.is_selected = true;
  587. }, this);
  588. } else {
  589. jsonData.items.forEach(function (service) {
  590. if (savedSelectedServices.contains(service.StackServices.service_name))
  591. service.StackServices.is_selected = true;
  592. else
  593. service.StackServices.is_selected = false;
  594. }, this);
  595. }
  596. if (!savedInstalledServices) {
  597. jsonData.items.forEach(function (service) {
  598. service.StackServices.is_installed = false;
  599. }, this);
  600. } else {
  601. jsonData.items.forEach(function (service) {
  602. if (savedInstalledServices.contains(service.StackServices.service_name))
  603. service.StackServices.is_installed = true;
  604. else
  605. service.StackServices.is_installed = false;
  606. }, this);
  607. }
  608. App.stackServiceMapper.mapStackServices(jsonData);
  609. },
  610. loadServiceComponentsErrorCallback: function (request, ajaxOptions, error) {
  611. },
  612. /**
  613. * Load config groups from local DB
  614. */
  615. loadServiceConfigGroups: function () {
  616. var props = this.getDBProperties(['serviceConfigGroups', 'hosts']);
  617. var serviceConfigGroups = props.serviceConfigGroups,
  618. hosts = props.hosts || {},
  619. host_names = Em.keys(hosts);
  620. if (Em.isNone(serviceConfigGroups)) {
  621. serviceConfigGroups = [];
  622. }
  623. else {
  624. serviceConfigGroups.forEach(function(group) {
  625. var hostNames = group.hosts.map(function(host_id) {
  626. for (var i = 0; i < host_names.length; i++) {
  627. if (hosts[host_names[i]].id === host_id) {
  628. return host_names[i];
  629. }
  630. }
  631. Em.assert('host is missing!!!!', false);
  632. });
  633. Em.set(group, 'hosts', hostNames);
  634. });
  635. }
  636. this.set('content.configGroups', serviceConfigGroups);
  637. },
  638. registerErrPopup: function (header, message) {
  639. App.ModalPopup.show({
  640. header: header,
  641. secondary: false,
  642. bodyClass: Ember.View.extend({
  643. template: Ember.Handlebars.compile('<p>{{view.message}}</p>'),
  644. message: message
  645. })
  646. });
  647. },
  648. /**
  649. * Save hosts that the user confirmed to proceed with from step 3
  650. * @param stepController App.WizardStep3Controller
  651. */
  652. saveConfirmedHosts: function (stepController) {
  653. var hosts = this.get('content.hosts'),
  654. indx = 1;
  655. //add previously installed hosts
  656. for (var hostName in hosts) {
  657. if (!hosts[hostName].isInstalled) {
  658. delete hosts[hostName];
  659. }
  660. }
  661. stepController.get('confirmedHosts').forEach(function (_host) {
  662. if (_host.bootStatus == 'REGISTERED') {
  663. hosts[_host.name] = {
  664. name: _host.name,
  665. cpu: _host.cpu,
  666. memory: _host.memory,
  667. disk_info: _host.disk_info,
  668. os_type: _host.os_type,
  669. os_arch: _host.os_arch,
  670. ip: _host.ip,
  671. bootStatus: _host.bootStatus,
  672. isInstalled: false,
  673. id: indx++
  674. };
  675. }
  676. });
  677. this.setDBProperty('hosts', hosts);
  678. this.set('content.hosts', hosts);
  679. },
  680. /**
  681. * Save data after installation to main controller
  682. * @param stepController App.WizardStep9Controller
  683. */
  684. saveInstalledHosts: function (stepController) {
  685. var hosts = stepController.get('hosts');
  686. var hostInfo = this.getDBProperty('hosts');
  687. for (var index in hostInfo) {
  688. hostInfo[index].status = "pending";
  689. var host = hosts.findProperty('name', hostInfo[index].name);
  690. if (host) {
  691. hostInfo[index].status = host.status;
  692. hostInfo[index].message = host.message;
  693. hostInfo[index].progress = host.progress;
  694. }
  695. }
  696. this.set('content.hosts', hostInfo);
  697. this.setDBProperty('hosts', hostInfo);
  698. },
  699. /**
  700. * Save slaveHostComponents to main controller
  701. * @param stepController
  702. */
  703. saveSlaveComponentHosts: function (stepController) {
  704. var hosts = stepController.get('hosts'),
  705. dbHosts = this.getDBProperty('hosts'),
  706. headers = stepController.get('headers');
  707. var formattedHosts = Ember.Object.create();
  708. headers.forEach(function (header) {
  709. formattedHosts.set(header.get('name'), []);
  710. });
  711. hosts.forEach(function (host) {
  712. var checkboxes = host.checkboxes;
  713. headers.forEach(function (header) {
  714. var cb = checkboxes.findProperty('title', header.get('label'));
  715. if (cb.checked) {
  716. formattedHosts.get(header.get('name')).push({
  717. group: 'Default',
  718. isInstalled: cb.isInstalled,
  719. host_id: dbHosts[host.hostName].id
  720. });
  721. }
  722. });
  723. });
  724. var slaveComponentHosts = [];
  725. headers.forEach(function (header) {
  726. slaveComponentHosts.push({
  727. componentName: header.get('name'),
  728. displayName: header.get('label').replace(/\s/g, ''),
  729. hosts: formattedHosts.get(header.get('name'))
  730. });
  731. });
  732. this.setDBProperty('slaveComponentHosts', slaveComponentHosts);
  733. this.set('content.slaveComponentHosts', slaveComponentHosts);
  734. },
  735. /**
  736. * Return true if cluster data is loaded and false otherwise.
  737. * This is used for all wizard controllers except for installer wizard.
  738. */
  739. dataLoading: function () {
  740. var dfd = $.Deferred();
  741. this.connectOutlet('loading');
  742. if (App.router.get('clusterController.isLoaded')) {
  743. dfd.resolve();
  744. } else {
  745. var interval = setInterval(function () {
  746. if (App.router.get('clusterController.isLoaded')) {
  747. dfd.resolve();
  748. clearInterval(interval);
  749. }
  750. }, 50);
  751. }
  752. return dfd.promise();
  753. },
  754. /**
  755. * Return true if user data is loaded via App.MainServiceInfoConfigsController
  756. * This function is used in reassign master wizard right now.
  757. */
  758. usersLoading: function () {
  759. var self = this;
  760. var dfd = $.Deferred();
  761. var miscController = App.MainAdminServiceAccountsController.create({content: self.get('content')});
  762. miscController.loadUsers();
  763. var interval = setInterval(function () {
  764. if (miscController.get('dataIsLoaded')) {
  765. if (self.get("content.hdfsUser")) {
  766. self.set('content.hdfsUser', miscController.get('content.hdfsUser'));
  767. }
  768. dfd.resolve();
  769. clearInterval(interval);
  770. }
  771. }, 10);
  772. return dfd.promise();
  773. },
  774. /**
  775. * Save cluster status before going to deploy step
  776. * @param name cluster state. Unique for every wizard
  777. */
  778. saveClusterState: function (name) {
  779. App.clusterStatus.setClusterStatus({
  780. clusterName: this.get('content.cluster.name'),
  781. clusterState: name,
  782. wizardControllerName: this.get('content.controllerName'),
  783. localdb: App.db.data
  784. });
  785. },
  786. /**
  787. * Load serviceConfigProperties to model
  788. */
  789. loadServiceConfigProperties: function () {
  790. var serviceConfigProperties = this.getDBProperty('serviceConfigProperties');
  791. this.set('content.serviceConfigProperties', serviceConfigProperties);
  792. },
  793. /**
  794. * Save config properties
  795. * @param stepController Step7WizardController
  796. */
  797. saveServiceConfigProperties: function (stepController) {
  798. var serviceConfigProperties = [];
  799. var fileNamesToUpdate = this.getDBProperty('fileNamesToUpdate') || [];
  800. var installedServiceNames = stepController.get('installedServiceNames') || [];
  801. var installedServiceNamesMap = installedServiceNames.toWickMap();
  802. stepController.get('stepConfigs').forEach(function (_content) {
  803. if (_content.serviceName === 'YARN') {
  804. _content.set('configs', App.config.textareaIntoFileConfigs(_content.get('configs'), 'capacity-scheduler.xml'));
  805. }
  806. _content.get('configs').forEach(function (_configProperties) {
  807. if (!Em.isNone(_configProperties.get('group'))) {
  808. return false;
  809. }
  810. var configProperty = App.config.createDefaultConfig(
  811. _configProperties.get('name'),
  812. _configProperties.get('serviceName'),
  813. _configProperties.get('filename'),
  814. _configProperties.get('isUserProperty'),
  815. {value: _configProperties.get('value')}
  816. );
  817. configProperty = App.config.mergeStaticProperties(configProperty, _configProperties, ['name', 'filename']);
  818. if (this.isExcludedConfig(configProperty)) {
  819. configProperty.value = '';
  820. }
  821. serviceConfigProperties.push(configProperty);
  822. }, this);
  823. // check for configs that need to update for installed services
  824. if (installedServiceNamesMap[_content.get('serviceName')]) {
  825. // get only modified configs
  826. var configs = _content.get('configs').filter(function (config) {
  827. if (config.get('isNotDefaultValue') || (config.get('savedValue') === null)) {
  828. return config.isRequiredByAgent!== false;
  829. }
  830. return false;
  831. });
  832. // if modified configs detected push all service's configs for update
  833. if (configs.length) {
  834. fileNamesToUpdate = fileNamesToUpdate.concat(configs.mapProperty('filename').uniq());
  835. }
  836. }
  837. }, this);
  838. this.setDBProperties({
  839. fileNamesToUpdate: fileNamesToUpdate,
  840. serviceConfigProperties: serviceConfigProperties
  841. });
  842. this.set('content.serviceConfigProperties', serviceConfigProperties);
  843. },
  844. isExcludedConfig: function (configProperty) {
  845. return this.get('sensibleConfigs').mapProperty('name').indexOf(configProperty.name) > -1
  846. && this.get('sensibleConfigs').mapProperty('filename').indexOf(configProperty.filename) > -1;
  847. },
  848. /**
  849. * save Config groups
  850. * @param stepController
  851. * @param isAddService
  852. */
  853. saveServiceConfigGroups: function (stepController, isAddService) {
  854. var serviceConfigGroups = [],
  855. isForInstalledService = false,
  856. hosts = isAddService ? App.router.get('addServiceController').getDBProperty('hosts') : this.getDBProperty('hosts');
  857. stepController.get('stepConfigs').forEach(function (service) {
  858. // mark group of installed service
  859. if (service.get('selected') === false) isForInstalledService = true;
  860. service.get('configGroups').forEach(function (configGroup) {
  861. var properties = [];
  862. configGroup.get('properties').forEach(function (property) {
  863. properties.push({
  864. isRequiredByAgent: property.get('isRequiredByAgent'),
  865. name: property.get('name'),
  866. value: property.get('value'),
  867. isFinal: property.get('isFinal'),
  868. filename: property.get('filename')
  869. })
  870. });
  871. //configGroup copied into plain JS object to avoid Converting circular structure to JSON
  872. var hostNames = configGroup.get('hosts').map(function(host_name) {return hosts[host_name].id;});
  873. serviceConfigGroups.push({
  874. id: configGroup.get('id'),
  875. name: configGroup.get('name'),
  876. description: configGroup.get('description'),
  877. hosts: hostNames.slice(),
  878. properties: properties.slice(),
  879. is_default: configGroup.get('isDefault'),
  880. is_for_installed_service: isForInstalledService,
  881. is_for_update: configGroup.isForUpdate || configGroup.get('hash') != this.getConfigGroupHash(configGroup, hostNames),
  882. service_name: configGroup.get('serviceName'),
  883. service_id: configGroup.get('serviceName'),
  884. desired_configs: configGroup.get('desiredConfigs'),
  885. config_group_id: configGroup.get('configGroupId'),
  886. child_config_groups: configGroup.get('childConfigGroups') ? configGroup.get('childConfigGroups').mapProperty('id') : [],
  887. parent_config_group_id: configGroup.get('parentConfigGroup.id')
  888. });
  889. }, this)
  890. }, this);
  891. this.setDBProperty('serviceConfigGroups', serviceConfigGroups);
  892. this.set('content.configGroups', serviceConfigGroups);
  893. },
  894. /**
  895. * generate string hash for config group
  896. * @param {Object} configGroup
  897. * @param {Array|undefined} hosts
  898. * @returns {String|null}
  899. * @method getConfigGroupHash
  900. */
  901. getConfigGroupHash: function(configGroup, hosts) {
  902. if (!Em.get(configGroup, 'properties.length') && !Em.get(configGroup, 'hosts.length') && !hosts) {
  903. return null;
  904. }
  905. var hash = {};
  906. Em.get(configGroup, 'properties').forEach(function (config) {
  907. hash[Em.get(config, 'name')] = {value: Em.get(config, 'value'), isFinal: Em.get(config, 'isFinal')};
  908. });
  909. hash['hosts'] = hosts || Em.get(configGroup, 'hosts');
  910. return JSON.stringify(hash);
  911. },
  912. /**
  913. * return slaveComponents bound to hosts
  914. * @return {Array}
  915. */
  916. getSlaveComponentHosts: function () {
  917. var components = this.get('slaveComponents');
  918. var result = [];
  919. var installedServices = App.Service.find().mapProperty('serviceName');
  920. var selectedServices = App.StackService.find().filterProperty('isSelected', true).mapProperty('serviceName');
  921. var installedComponentsMap = {};
  922. var uninstalledComponents = [];
  923. components.forEach(function (component) {
  924. if (installedServices.contains(component.get('serviceName'))) {
  925. installedComponentsMap[component.get('componentName')] = [];
  926. } else if (selectedServices.contains(component.get('serviceName'))) {
  927. uninstalledComponents.push(component);
  928. }
  929. }, this);
  930. installedComponentsMap['HDFS_CLIENT'] = [];
  931. App.HostComponent.find().forEach(function (hostComponent) {
  932. if (installedComponentsMap[hostComponent.get('componentName')]) {
  933. installedComponentsMap[hostComponent.get('componentName')].push(hostComponent.get('hostName'));
  934. }
  935. }, this);
  936. for (var componentName in installedComponentsMap) {
  937. var name = (componentName === 'HDFS_CLIENT') ? 'CLIENT' : componentName;
  938. var component = {
  939. componentName: name,
  940. displayName: App.format.role(name),
  941. hosts: [],
  942. isInstalled: true
  943. };
  944. installedComponentsMap[componentName].forEach(function (hostName) {
  945. component.hosts.push({
  946. group: "Default",
  947. hostName: hostName,
  948. isInstalled: true
  949. });
  950. }, this);
  951. result.push(component);
  952. }
  953. uninstalledComponents.forEach(function (component) {
  954. var hosts = jQuery.extend(true, [], result.findProperty('componentName', 'DATANODE').hosts);
  955. hosts.setEach('isInstalled', false);
  956. result.push({
  957. componentName: component.get('componentName'),
  958. displayName: App.format.role(component.get('componentName')),
  959. hosts: hosts,
  960. isInstalled: false
  961. })
  962. });
  963. return result;
  964. },
  965. /**
  966. * Load master component hosts data for using in required step controllers
  967. */
  968. loadMasterComponentHosts: function () {
  969. var masterComponentHosts = this.getDBProperty('masterComponentHosts');
  970. var stackMasterComponents = App.get('components.masters').uniq();
  971. if (!masterComponentHosts) {
  972. masterComponentHosts = [];
  973. App.HostComponent.find().filter(function(component) {
  974. return stackMasterComponents.contains(component.get('componentName'));
  975. }).forEach(function (item) {
  976. masterComponentHosts.push({
  977. component: item.get('componentName'),
  978. hostName: item.get('hostName'),
  979. isInstalled: true,
  980. serviceId: item.get('service.id'),
  981. display_name: item.get('displayName')
  982. })
  983. });
  984. this.setDBProperty('masterComponentHosts', masterComponentHosts);
  985. }
  986. this.set("content.masterComponentHosts", masterComponentHosts);
  987. },
  988. /**
  989. * Save Master Component Hosts data to Main Controller
  990. * @param stepController App.WizardStep5Controller
  991. */
  992. saveMasterComponentHosts: function (stepController) {
  993. var obj = stepController.get('selectedServicesMasters');
  994. var masterComponentHosts = [];
  995. obj.forEach(function (_component) {
  996. masterComponentHosts.push({
  997. display_name: _component.get('display_name'),
  998. component: _component.get('component_name'),
  999. hostName: _component.get('selectedHost'),
  1000. serviceId: _component.get('serviceId'),
  1001. isInstalled: _component.get('isInstalled')
  1002. });
  1003. });
  1004. this.setDBProperty('masterComponentHosts', masterComponentHosts);
  1005. this.set('content.masterComponentHosts', masterComponentHosts);
  1006. },
  1007. clearMasterComponentHosts: function() {
  1008. this.set('content.masterComponentHosts', null);
  1009. this.setDBProperty('masterComponentHosts', null);
  1010. },
  1011. /**
  1012. * Load information about hosts with clients components
  1013. */
  1014. loadClients: function () {
  1015. var clients = this.getDBProperty('clientInfo');
  1016. this.set('content.clients', clients);
  1017. },
  1018. /**
  1019. * load methods assigned to each step
  1020. * methods executed in exact order as they described in map
  1021. * @return {object}
  1022. */
  1023. loadAllPriorSteps: function () {
  1024. var currentStep = this.get('currentStep');
  1025. var loadMap = this.get('loadMap');
  1026. var operationStack = [];
  1027. var dfd = $.Deferred();
  1028. for (var s in loadMap) {
  1029. if (parseInt(s) <= parseInt(currentStep)) {
  1030. operationStack.pushObjects(loadMap[s]);
  1031. }
  1032. }
  1033. var sequence = App.actionSequence.create({context: this});
  1034. sequence.setSequence(operationStack).onFinish(function () {
  1035. dfd.resolve();
  1036. }).start();
  1037. return dfd.promise();
  1038. },
  1039. /**
  1040. * return new object extended from clusterStatusTemplate
  1041. * @return Object
  1042. */
  1043. getCluster: function () {
  1044. return jQuery.extend({}, this.get('clusterStatusTemplate'), {name: App.router.getClusterName()});
  1045. },
  1046. /**
  1047. * Load services data from server.
  1048. */
  1049. loadServicesFromServer: function () {
  1050. var services = this.getDBProperty('services');
  1051. if (!services) {
  1052. services = {
  1053. selectedServices: [],
  1054. installedServices: []
  1055. };
  1056. App.StackService.find().forEach(function(item){
  1057. var isInstalled = App.Service.find().someProperty('id', item.get('serviceName'));
  1058. item.set('isSelected', isInstalled);
  1059. item.set('isInstalled', isInstalled);
  1060. if (isInstalled) {
  1061. services.selectedServices.push(item.get('serviceName'));
  1062. services.installedServices.push(item.get('serviceName'));
  1063. }
  1064. },this);
  1065. this.setDBProperty('services',services);
  1066. } else {
  1067. App.StackService.find().forEach(function(item) {
  1068. var isSelected = services.selectedServices.contains(item.get('serviceName'));
  1069. var isInstalled = services.installedServices.contains(item.get('serviceName'));
  1070. item.set('isSelected', isSelected);
  1071. item.set('isInstalled', isInstalled);
  1072. },this);
  1073. }
  1074. this.set('content.services', App.StackService.find());
  1075. },
  1076. /**
  1077. * Load confirmed hosts.
  1078. * Will be used at <code>Assign Masters(step5)</code> step
  1079. */
  1080. loadConfirmedHosts: function () {
  1081. var hosts = App.db.getHosts();
  1082. if (hosts) {
  1083. this.set('content.hosts', hosts);
  1084. }
  1085. },
  1086. loadHosts: function () {
  1087. var dfd;
  1088. var hostsInDb = this.getDBProperty('hosts');
  1089. if (hostsInDb) {
  1090. this.set('content.hosts', hostsInDb);
  1091. dfd = $.Deferred();
  1092. dfd.resolve();
  1093. } else {
  1094. dfd = App.ajax.send({
  1095. name: 'hosts.confirmed',
  1096. sender: this,
  1097. data: {},
  1098. success: 'loadHostsSuccessCallback',
  1099. error: 'loadHostsErrorCallback'
  1100. });
  1101. }
  1102. return dfd.promise();
  1103. },
  1104. loadHostsSuccessCallback: function (response) {
  1105. var installedHosts = {};
  1106. response.items.forEach(function (item, indx) {
  1107. installedHosts[item.Hosts.host_name] = {
  1108. name: item.Hosts.host_name,
  1109. cpu: item.Hosts.cpu_count,
  1110. memory: item.Hosts.total_mem,
  1111. disk_info: item.Hosts.disk_info,
  1112. osType: item.Hosts.os_type,
  1113. osArch: item.Hosts.os_arch,
  1114. ip: item.Hosts.ip,
  1115. bootStatus: "REGISTERED",
  1116. isInstalled: true,
  1117. hostComponents: item.host_components,
  1118. id: indx++
  1119. };
  1120. });
  1121. this.setDBProperty('hosts', installedHosts);
  1122. this.set('content.hosts', installedHosts);
  1123. },
  1124. loadHostsErrorCallback: function (jqXHR, ajaxOptions, error, opt) {
  1125. App.ajax.defaultErrorHandler(jqXHR, opt.url, opt.method, jqXHR.status);
  1126. },
  1127. /**
  1128. * Determine if <code>Assign Slaves and Clients</code> step should be skipped
  1129. * @method setSkipSlavesStep
  1130. * @param services
  1131. * @param step
  1132. */
  1133. setSkipSlavesStep: function (services, step) {
  1134. var hasServicesWithSlave = services.someProperty('hasSlave');
  1135. var hasServicesWithClient = services.someProperty('hasClient');
  1136. var hasServicesWithCustomAssignedNonMasters = services.someProperty('hasNonMastersWithCustomAssignment');
  1137. this.set('content.skipSlavesStep', !hasServicesWithSlave && !hasServicesWithClient || !hasServicesWithCustomAssignedNonMasters);
  1138. if (this.get('content.skipSlavesStep')) {
  1139. this.get('isStepDisabled').findProperty('step', step).set('value', this.get('content.skipSlavesStep'));
  1140. }
  1141. },
  1142. /**
  1143. * Load config themes for enhanced config layout.
  1144. *
  1145. * @method loadConfigThemes
  1146. * @return {$.Deferred}
  1147. */
  1148. loadConfigThemes: function () {
  1149. var self = this;
  1150. var dfd = $.Deferred();
  1151. if (!this.get('stackConfigsLoaded')) {
  1152. var serviceNames = App.StackService.find().filter(function (s) {
  1153. return s.get('isSelected') || s.get('isInstalled');
  1154. }).mapProperty('serviceName');
  1155. // Load stack configs before loading themes
  1156. App.config.loadClusterConfigsFromStack().always(function() {
  1157. App.config.loadConfigsFromStack(serviceNames).done(function () {
  1158. if (App.get('isClusterSupportsEnhancedConfigs')) {
  1159. self.loadConfigThemeForServices(serviceNames).always(function () {
  1160. self.set('stackConfigsLoaded', true);
  1161. App.themesMapper.generateAdvancedTabs(serviceNames);
  1162. dfd.resolve();
  1163. });
  1164. } else {
  1165. self.set('stackConfigsLoaded', true);
  1166. dfd.resolve();
  1167. }
  1168. });
  1169. });
  1170. }
  1171. else {
  1172. dfd.resolve();
  1173. this.set('stackConfigsLoaded', true);
  1174. }
  1175. return dfd.promise();
  1176. },
  1177. /**
  1178. * Cache "stepConfigs" to local storage in name value pairs
  1179. * @param stepController
  1180. */
  1181. cacheStepConfigValues: function(stepController) {
  1182. var self = this;
  1183. var stepConfigs = [];
  1184. stepController.get("stepConfigs").forEach(function (category) {
  1185. var configs = category.configs.map(function(config) {
  1186. if (self.isExcludedConfig(config)) {
  1187. config.set('value', '');
  1188. }
  1189. return {
  1190. name: config.name,
  1191. filename: config.filename,
  1192. value: config.value
  1193. };
  1194. });
  1195. stepConfigs = stepConfigs.concat(configs);
  1196. });
  1197. if (stepConfigs.length > 0 ) {
  1198. this.setDBProperty(stepController.name + "-sc", stepConfigs);
  1199. }
  1200. },
  1201. loadCachedStepConfigValues: function(stepController) {
  1202. return this.getDBProperty(stepController.name + "-sc");
  1203. },
  1204. clearCachedStepConfigValues: function(stepController) {
  1205. this.setDBProperty(stepController.name + "-sc", null);
  1206. },
  1207. saveTasksStatuses: function (tasksStatuses) {
  1208. this.set('content.tasksStatuses', tasksStatuses);
  1209. this.setDBProperty('tasksStatuses', tasksStatuses);
  1210. },
  1211. loadTasksStatuses: function() {
  1212. var tasksStatuses = this.getDBProperty('tasksStatuses');
  1213. this.set('content.tasksStatuses', tasksStatuses);
  1214. },
  1215. saveTasksRequestIds: function (tasksRequestIds) {
  1216. this.set('content.tasksRequestIds', tasksRequestIds);
  1217. this.setDBProperty('tasksRequestIds', tasksRequestIds);
  1218. },
  1219. loadTasksRequestIds: function() {
  1220. var tasksRequestIds = this.getDBProperty('tasksRequestIds');
  1221. this.set('content.tasksRequestIds', tasksRequestIds);
  1222. },
  1223. saveRequestIds: function (requestIds) {
  1224. this.set('content.requestIds', requestIds);
  1225. this.setDBProperty('requestIds', requestIds);
  1226. },
  1227. loadRequestIds: function() {
  1228. var requestIds = this.getDBProperty('requestIds');
  1229. this.set('content.requestIds', requestIds);
  1230. },
  1231. loadRecommendations: function () {
  1232. this.set("content.recommendations", this.getDBProperty('recommendations'));
  1233. }
  1234. });