import Ember from 'ember'; import base64js from 'base64-js'; const { Component, inject, computed, get } = Ember; const { camelize } = Ember.String; const DEFAULTS = { key: null, loading: false, errors: [], threshold: null, progress: null, pgp_key: null, haveSavedPGPKey: false, started: false, generateWithPGP: false, pgpKeyFile: { value: '' }, nonce: '', }; export default Component.extend(DEFAULTS, { tagName: '', store: inject.service(), formText: null, fetchOnInit: false, buttonText: 'Submit', thresholdPath: 'required', generateAction: false, encoded_token: null, init() { if (this.get('fetchOnInit')) { this.attemptProgress(); } return this._super(...arguments); }, onShamirSuccess() {}, // can be overridden w/an attr isComplete(data) { return data.complete === true; }, stopLoading() { this.setProperties({ loading: false, errors: [], key: null, }); }, reset() { this.setProperties(DEFAULTS); }, hasProgress: computed.gt('progress', 0), actionSuccess(resp) { const { isComplete, onShamirSuccess, thresholdPath } = this.getProperties( 'isComplete', 'onShamirSuccess', 'thresholdPath' ); this.stopLoading(); this.set('threshold', get(resp, thresholdPath)); this.setProperties(resp); if (isComplete(resp)) { this.reset(); onShamirSuccess(resp); } }, actionError(e) { this.stopLoading(); if (e.httpStatus === 400) { this.set('errors', e.errors); } else { throw e; } }, generateStep: computed('generateWithPGP', 'haveSavedPGPKey', 'otp', 'pgp_key', function() { let { generateWithPGP, otp, pgp_key, haveSavedPGPKey } = this.getProperties( 'generateWithPGP', 'otp', 'pgp_key', 'haveSavedPGPKey' ); if (!generateWithPGP && !pgp_key && !otp) { return 'chooseMethod'; } if (otp) { return 'beginGenerationWithOTP'; } if (generateWithPGP) { if (pgp_key && haveSavedPGPKey) { return 'beginGenerationWithPGP'; } else { return 'providePGPKey'; } } }), extractData(data) { const isGenerate = this.get('generateAction'); const hasStarted = this.get('started'); const usePGP = this.get('generateWithPGP'); const nonce = this.get('nonce'); if (!isGenerate || hasStarted) { if (nonce) { data.nonce = nonce; } return data; } if (usePGP) { return { pgp_key: data.pgp_key, }; } return { otp: data.otp, }; }, attemptProgress(data) { const checkStatus = data ? false : true; let action = this.get('action'); action = action && camelize(action); this.set('loading', true); const adapter = this.get('store').adapterFor('cluster'); const method = adapter[action]; method .call(adapter, data, { checkStatus }) .then(resp => this.actionSuccess(resp), (...args) => this.actionError(...args)); }, actions: { reset() { this.reset(); this.set('encoded_token', null); this.set('otp', null); }, onSubmit(data) { if (!data.key) { return; } this.attemptProgress(this.extractData(data)); }, startGenerate(data) { this.attemptProgress(this.extractData(data)); }, generateOTP() { const bytes = new window.Uint8Array(16); window.crypto.getRandomValues(bytes); this.set('otp', base64js.fromByteArray(bytes)); }, setKey(_, keyFile) { this.set('pgp_key', keyFile.value); this.set('pgpKeyFile', keyFile); }, savePGPKey() { if (this.get('pgp_key')) { this.set('haveSavedPGPKey', true); } }, }, });