|
| 1 | +'use strict'; |
| 2 | + |
| 3 | +/** |
| 4 | + * @ngdoc function |
| 5 | + * @name openshiftConsole.controller:AddConfigVolumeController |
| 6 | + * @description |
| 7 | + * # AddConfigVolumeController |
| 8 | + * Controller of the openshiftConsole |
| 9 | + */ |
| 10 | +angular.module('openshiftConsole') |
| 11 | + .controller('AddConfigVolumeController', |
| 12 | + function($filter, |
| 13 | + $routeParams, |
| 14 | + $scope, |
| 15 | + $window, |
| 16 | + APIService, |
| 17 | + BreadcrumbsService, |
| 18 | + DataService, |
| 19 | + Navigate, |
| 20 | + ProjectsService, |
| 21 | + StorageService) { |
| 22 | + if (!$routeParams.kind || !$routeParams.name) { |
| 23 | + Navigate.toErrorPage("Kind or name parameter missing."); |
| 24 | + return; |
| 25 | + } |
| 26 | + |
| 27 | + var supportedKinds = [ |
| 28 | + 'Deployment', |
| 29 | + 'DeploymentConfig', |
| 30 | + 'ReplicaSet', |
| 31 | + 'ReplicationController' |
| 32 | + ]; |
| 33 | + |
| 34 | + if (!_.includes(supportedKinds, $routeParams.kind)) { |
| 35 | + Navigate.toErrorPage("Volumes are not supported for kind " + $routeParams.kind + "."); |
| 36 | + return; |
| 37 | + } |
| 38 | + |
| 39 | + var resourceGroupVersion = { |
| 40 | + resource: APIService.kindToResource($routeParams.kind), |
| 41 | + group: $routeParams.group |
| 42 | + }; |
| 43 | + |
| 44 | + $scope.alerts = {}; |
| 45 | + $scope.projectName = $routeParams.project; |
| 46 | + $scope.kind = $routeParams.kind; |
| 47 | + $scope.name = $routeParams.name; |
| 48 | + $scope.attach = { |
| 49 | + allContainers: true, |
| 50 | + pickKeys: false |
| 51 | + }; |
| 52 | + $scope.forms = {}; |
| 53 | + |
| 54 | + $scope.breadcrumbs = BreadcrumbsService.getBreadcrumbs({ |
| 55 | + name: $routeParams.name, |
| 56 | + kind: $routeParams.kind, |
| 57 | + namespace: $routeParams.project, |
| 58 | + subpage: 'Add Config Files', |
| 59 | + includeProject: true |
| 60 | + }); |
| 61 | + |
| 62 | + var humanizeKind = $filter('humanizeKind'); |
| 63 | + $scope.groupByKind = function(object) { |
| 64 | + return humanizeKind(object.kind); |
| 65 | + }; |
| 66 | + |
| 67 | + var resetItems = function() { |
| 68 | + // Add an empty item so one appears when the section is first expanded. |
| 69 | + _.set($scope, 'attach.items', [{}]); |
| 70 | + }; |
| 71 | + |
| 72 | + // Clear the items if the source has changed. |
| 73 | + $scope.$watch('attach.source', resetItems); |
| 74 | + |
| 75 | + var setDirty = function() { |
| 76 | + $scope.forms.addConfigVolumeForm.$setDirty(); |
| 77 | + }; |
| 78 | + |
| 79 | + $scope.addItem = function() { |
| 80 | + $scope.attach.items.push({}); |
| 81 | + setDirty(); |
| 82 | + }; |
| 83 | + |
| 84 | + $scope.removeItem = function(index) { |
| 85 | + $scope.attach.items.splice(index, 1); |
| 86 | + setDirty(); |
| 87 | + }; |
| 88 | + |
| 89 | + ProjectsService |
| 90 | + .get($routeParams.project) |
| 91 | + .then(_.spread(function(project, context) { |
| 92 | + $scope.project = project; |
| 93 | + |
| 94 | + var orderByDisplayName = $filter('orderByDisplayName'); |
| 95 | + var getErrorDetails = $filter('getErrorDetails'); |
| 96 | + var generateName = $filter('generateName'); |
| 97 | + |
| 98 | + var displayError = function(errorMessage, errorDetails) { |
| 99 | + $scope.disableInputs = true; |
| 100 | + $scope.alerts['attach-persistent-volume-claim'] = { |
| 101 | + type: "error", |
| 102 | + message: errorMessage, |
| 103 | + details: errorDetails |
| 104 | + }; |
| 105 | + }; |
| 106 | + |
| 107 | + DataService.get(resourceGroupVersion, $routeParams.name, context, { errorNotification: false }).then( |
| 108 | + function(object) { |
| 109 | + $scope.targetObject = object; |
| 110 | + $scope.breadcrumbs = BreadcrumbsService.getBreadcrumbs({ |
| 111 | + object: object, |
| 112 | + project: project, |
| 113 | + subpage: 'Add Config Files', |
| 114 | + includeProject: true |
| 115 | + }); |
| 116 | + }, |
| 117 | + function(e) { |
| 118 | + $scope.error = e; |
| 119 | + } |
| 120 | + ); |
| 121 | + |
| 122 | + DataService.list("configmaps", context, null, { errorNotification: false }).then(function(configMapData) { |
| 123 | + $scope.configMaps = orderByDisplayName(configMapData.by("metadata.name")); |
| 124 | + }, function(e) { |
| 125 | + if (e.status === 403) { |
| 126 | + $scope.configMaps = []; |
| 127 | + return; |
| 128 | + } |
| 129 | + |
| 130 | + displayError('Could not load config maps', getErrorDetails(e)); |
| 131 | + }); |
| 132 | + DataService.list("secrets", context, null, { errorNotification: false }).then(function(secretData) { |
| 133 | + $scope.secrets = orderByDisplayName(secretData.by("metadata.name")); |
| 134 | + }, function(e) { |
| 135 | + if (e.status === 403) { |
| 136 | + $scope.secrets = []; |
| 137 | + return; |
| 138 | + } |
| 139 | + |
| 140 | + displayError('Could not load secrets', getErrorDetails(e)); |
| 141 | + }); |
| 142 | + |
| 143 | + var isContainerSelected = function(container) { |
| 144 | + return $scope.attach.allContainers || $scope.attach.containers[container.name]; |
| 145 | + }; |
| 146 | + |
| 147 | + // Look at the existing mount paths so that we can warn if the new value is not unique. |
| 148 | + var updateMountPaths = function() { |
| 149 | + var podTemplate = _.get($scope, 'targetObject.spec.template'); |
| 150 | + $scope.existingMountPaths = StorageService.getMountPaths(podTemplate, isContainerSelected); |
| 151 | + }; |
| 152 | + |
| 153 | + $scope.$watchGroup(['targetObject', 'attach.allContainers'], updateMountPaths); |
| 154 | + $scope.$watch('attach.containers', updateMountPaths, true); |
| 155 | + |
| 156 | + // Make sure the path for each item is unique. |
| 157 | + var updateItemPaths = function() { |
| 158 | + // Call `_.compact` to remove empty values. |
| 159 | + var paths = _.map($scope.attach.items, 'path'); |
| 160 | + $scope.itemPaths = _.compact(paths); |
| 161 | + }; |
| 162 | + $scope.$watch('attach.items', updateItemPaths, true); |
| 163 | + |
| 164 | + $scope.addVolume = function() { |
| 165 | + if ($scope.forms.addConfigVolumeForm.$invalid) { |
| 166 | + return; |
| 167 | + } |
| 168 | + |
| 169 | + var resource = $scope.targetObject; |
| 170 | + var source = _.get($scope, 'attach.source'); |
| 171 | + var podTemplate = _.get(resource, 'spec.template'); |
| 172 | + var name = generateName('volume-'); |
| 173 | + var mountPath = _.get($scope, 'attach.mountPath'); |
| 174 | + var newVolumeMount = { |
| 175 | + name: name, |
| 176 | + mountPath: mountPath |
| 177 | + }; |
| 178 | + |
| 179 | + // The volume mount is read-only for secrets. |
| 180 | + if (source.kind === 'Secret') { |
| 181 | + newVolumeMount.readOnly = true; |
| 182 | + } |
| 183 | + |
| 184 | + // For each selected container, add the new volume mount. |
| 185 | + _.each(podTemplate.spec.containers, function(container) { |
| 186 | + if (isContainerSelected(container)) { |
| 187 | + container.volumeMounts = container.volumeMounts || []; |
| 188 | + container.volumeMounts.push(newVolumeMount); |
| 189 | + } |
| 190 | + }); |
| 191 | + |
| 192 | + var newVolume = { |
| 193 | + name: name |
| 194 | + }; |
| 195 | + |
| 196 | + var items; |
| 197 | + if ($scope.attach.pickKeys) { |
| 198 | + items = $scope.attach.items; |
| 199 | + } |
| 200 | + |
| 201 | + switch (source.kind) { |
| 202 | + case 'ConfigMap': |
| 203 | + newVolume.configMap = { |
| 204 | + name: source.metadata.name, |
| 205 | + items: items |
| 206 | + }; |
| 207 | + break; |
| 208 | + case 'Secret': |
| 209 | + newVolume.secret = { |
| 210 | + secretName: source.metadata.name, |
| 211 | + items: items |
| 212 | + }; |
| 213 | + break; |
| 214 | + } |
| 215 | + |
| 216 | + podTemplate.spec.volumes = podTemplate.spec.volumes || []; |
| 217 | + podTemplate.spec.volumes.push(newVolume); |
| 218 | + |
| 219 | + // Clear any previous alerts. |
| 220 | + $scope.alerts = {}; |
| 221 | + |
| 222 | + $scope.disableInputs = true; |
| 223 | + DataService.update(resourceGroupVersion, resource.metadata.name, $scope.targetObject, context).then( |
| 224 | + function() { |
| 225 | + $window.history.back(); |
| 226 | + }, |
| 227 | + function(result) { |
| 228 | + displayError("An error occurred attaching the persistent volume claim to the " + $filter('humanizeKind')($routeParams.kind) + ".", getErrorDetails(result)); |
| 229 | + } |
| 230 | + ); |
| 231 | + }; |
| 232 | + })); |
| 233 | + }); |
0 commit comments