import Component from '@ember/component' import ClusterDriver from 'shared/mixins/cluster-driver'; import { equal, alias } from '@ember/object/computed'; import { get, set, computed, observer } from '@ember/object'; import { satisfies } from 'shared/utils/parse-version'; import { inject as service } from '@ember/service'; import C from 'shared/utils/constants'; import layout from './template'; export default Component.extend(ClusterDriver, { layout, globalStore: service(), settings: service(), intl: service(), configField: 'rancherKubernetesEngineConfig', model: null, initialVersion: null, isEdit: equal('mode', 'edit'), clusterState: alias('model.originalCluster.state'), networkChoices: [ { label: 'clusterNew.rke.network.flannel', value: 'flannel' }, { label: 'clusterNew.rke.network.calico', value: 'calico' }, { label: 'clusterNew.rke.network.canal', value: 'canal' }, ], authChoices: [ { label: 'clusterNew.rke.auth.x509', value: 'x509' }, ], registry: 'default', registryUrl: null, registryUser: null, registryPass: null, init() { this._super(); const globalStore = get(this, 'globalStore'); const counts = {}; set(this, 'existingNodes', globalStore.all('node')); globalStore.findAll('node').then((all) => { all.forEach((node) => { const id = get(node,'clusterId'); counts[id] = (counts[id]||0) + 1; }); this.notifyPropertyChange('initialNodeCounts'); }); set(this, 'initialNodeCounts', counts); set(this, 'initialVersion', get(this, 'cluster.rancherKubernetesEngineConfig.kubernetesVersion')); }, didInsertElement() { if ( ! get(this,'cluster.rancherKubernetesEngineConfig') ) { const globalStore = get(this, 'globalStore'); let config = globalStore.createRecord({ type: 'rancherKubernetesEngineConfig', kubernetesVersion: get(this, `settings.${C.SETTING.VERSION_K8S_DEFAULT}`), authentication: globalStore.createRecord({ type: 'authnConfig', strategy: 'x509', }), network: globalStore.createRecord({ type: 'networkConfig', plugin: 'canal', }), services: globalStore.createRecord({ type: 'rkeConfigServices', kubeApi: globalStore.createRecord({ type: 'kubeAPIService', podSecurityPolicy: false, }), }), }); set(this, 'cluster.rancherKubernetesEngineConfig', config); } }, didReceiveAttrs() { if ( get(this,'isEdit') ) { this.loadToken(); } }, actions: { setNodePoolErrors(errors) { set(this, 'nodePoolErrors', errors); }, }, willSave() { if ( get(this, 'registry') === 'custom' ) { const registry = { url: get(this, 'registryUrl'), user: get(this, 'registryUser'), password: get(this, 'registryPass'), } set(this, 'config.privateRegistries', [registry]); } return this._super(...arguments); }, validate() { this._super(...arguments); const errors = get(this,'errors')||[]; if ( !get(this, 'isCustom') ) { errors.pushObjects(get(this, 'nodePoolErrors')); } if ( get(this, 'config.services.kubeApi.podSecurityPolicy') && !get(this, 'cluster.defaultPodSecurityPolicyTemplateId') ) { errors.push(get(this, 'intl').t('clusterNew.psp.required')); } set(this, 'errors', errors); return errors.length === 0; }, doneSaving() { if ( get(this, 'isCustom') ) { if ( get(this, 'isEdit') ) { this.sendAction('close'); } else { this.loadToken(); } } else { this.sendAction('close'); } }, loadToken() { const cluster = get(this,'cluster'); return cluster.getOrCreateToken().then((token) => { set(this, 'token', token); set(this, 'step', 2); }); }, versionChanged: observer('config.kubernetesVersion','versionChoices.[]', function() { const versions = get(this, 'versionChoices')||[]; const current = get(this, 'config.kubernetesVersion'); const exists = versions.findBy('value', current); if ( !exists ) { set(this, 'config.kubernetesVersion', versions[0].value); } }), versionChoices: computed('initialVersion', `settings.${C.SETTING.VERSIONS_K8S}`, 'config.kubernetesVersion', function() { const versions = JSON.parse(get(this, `settings.${C.SETTING.VERSIONS_K8S}`)||'{}'); if ( !versions ) { return []; } const initialVersion = get(this, 'initialVersion'); let oldestSupportedVersion = '>=1.8.0'; if ( initialVersion ) { oldestSupportedVersion = '>=' + initialVersion; } let out = Object.keys(versions); out = out.filter((v) => { const str = v.replace(/-.*/,''); return satisfies(str, oldestSupportedVersion); }); if (get(this, 'editing') && !out.includes(initialVersion) ) { out.unshift(initialVersion); } return out.map((v) => { return {value: v} }); }), // Custom stuff isCustom: equal('nodeWhich','custom'), existingNodes: null, initialNodeCounts: null, step: 1, token: null, labels: null, etcd: false, controlplane: false, worker: true, newNodeCount: computed('initialNodeCounts','cluster.id','existingNodes.@each.clusterId', function() { let clusterId = get(this,'requestedClusterId'); let orig = get(this, 'initialNodeCounts')[clusterId] || 0; let cur = get(this, 'existingNodes').filterBy('clusterId', clusterId).length if ( cur < orig ) { orig = cur; set (get(this,'initialNodeCounts'), clusterId, cur) } return cur - orig; }), command: computed(`settings.${C.SETTING.AGENT_IMAGE}`, 'labels', 'token.token', 'etcd', 'controlplane', 'worker', function() { const image = get(this,`settings.${C.SETTING.AGENT_IMAGE}`); const cacerts = get(this,`settings.${C.SETTING.CA_CERTS}`); const checksum = AWS.util.crypto.sha256(cacerts+'\n','hex'); const url = window.location.origin; const token = get(this,'token.token'); let roleFlags=''; const roles = ['etcd','controlplane','worker']; for ( let i = 0, k ; i < roles.length ; i++ ) { k = roles[i]; if ( get(this,k) ) { roleFlags += ' --' + k; } } let out = `docker run -d --restart=unless-stopped -v /var/run/docker.sock:/var/run/docker.sock --net=host ${image} --server ${url} --token ${token} --ca-checksum ${checksum}${roleFlags}`; const labels = get(this, 'labels')||{}; Object.keys(labels).forEach((key) => { out += ` --label ${key}=${labels[key]}`; }); return out; }), });