/** * Controller for network * * @module app/configuration * @exports networkController * @name networkController */ window.angular && (function(angular) { 'use strict'; angular.module('app.configuration').controller('networkController', [ '$scope', '$window', 'APIUtils', 'dataService', '$timeout', '$route', '$q', 'ngToast', function( $scope, $window, APIUtils, dataService, $timeout, $route, $q, ngToast) { $scope.dataService = dataService; $scope.network = {}; $scope.oldInterface = {}; $scope.interface = {}; $scope.networkDevice = false; $scope.hostname = ''; $scope.defaultGateway = ''; $scope.selectedInterface = ''; $scope.confirmSettings = false; $scope.loading = false; $scope.ipv4sToDelete = []; loadNetworkInfo(); $scope.selectInterface = function(interfaceId) { $scope.interface = $scope.network.interfaces[interfaceId]; // Copy the interface so we know later if changes were made to the page $scope.oldInterface = JSON.parse(JSON.stringify($scope.interface)); $scope.selectedInterface = interfaceId; $scope.networkDevice = false; }; $scope.addDNSField = function() { $scope.interface.Nameservers.push(''); }; $scope.removeDNSField = function(index) { $scope.interface.Nameservers.splice(index, 1); }; $scope.addIpv4Field = function() { $scope.interface.ipv4.values.push( {Address: '', PrefixLength: '', Gateway: ''}); }; $scope.removeIpv4Address = function(index) { // Check if the IPV4 being removed has an id. This indicates that it is // an existing address and needs to be removed in the back end. if ($scope.interface.ipv4.values[index].id) { $scope.ipv4sToDelete.push($scope.interface.ipv4.values[index]); } $scope.interface.ipv4.values.splice(index, 1); }; $scope.setNetworkSettings = function() { // Hides the confirm network settings modal $scope.confirmSettings = false; $scope.loading = true; var promises = []; // MAC Address are case-insensitive if ($scope.interface.MACAddress.toLowerCase() != dataService.mac_address.toLowerCase()) { promises.push(setMACAddress()); } if ($scope.defaultGateway != dataService.defaultgateway) { promises.push(setDefaultGateway()); } if ($scope.hostname != dataService.hostname) { promises.push(setHostname()); } if ($scope.interface.DHCPEnabled != $scope.oldInterface.DHCPEnabled) { promises.push(setDHCPEnabled()); } // Remove any empty strings from the array. Important because we add an // empty string to the end so the user can add a new DNS server, if the // user doesn't fill out the field, we don't want to add. $scope.interface.Nameservers = $scope.interface.Nameservers.filter(Boolean); // toString() is a cheap way to compare 2 string arrays if ($scope.interface.Nameservers.toString() != $scope.oldInterface.Nameservers.toString()) { promises.push(setNameservers()); } // Set IPV4 IP Addresses, Netmask Prefix Lengths, and Gateways if (!$scope.interface.DHCPEnabled) { // Delete existing IPV4 addresses that were removed promises.push(removeIPV4s()); // Update any changed IPV4 addresses and add new for (var i in $scope.interface.ipv4.values) { if (!APIUtils.validIPV4IP( $scope.interface.ipv4.values[i].Address)) { ngToast.danger( $scope.interface.ipv4.values[i].Address + ' invalid IP parameter'); $scope.loading = false; return; } if (!APIUtils.validIPV4IP( $scope.interface.ipv4.values[i].Gateway)) { ngToast.danger( $scope.interface.ipv4.values[i].Address + ' invalid gateway parameter'); $scope.loading = false; return; } // The netmask prefix length will be undefined if outside range if (!$scope.interface.ipv4.values[i].PrefixLength) { ngToast.danger( $scope.interface.ipv4.values[i].Address + ' invalid Prefix Length parameter'); $scope.loading = false; return; } if ($scope.interface.ipv4.values[i].updateAddress || $scope.interface.ipv4.values[i].updateGateway || $scope.interface.ipv4.values[i].updatePrefix) { // If IPV4 has an id it means it already exists in the back end, // and in order to update it is required to remove previous IPV4 // address and add new one. See openbmc/openbmc/issues/2163. // TODO: update to use PUT once issue 2163 is resolved. if ($scope.interface.ipv4.values[i].id) { promises.push(updateIPV4(i)); } else { promises.push(addIPV4(i)); } } } } if (promises.length) { $q.all(promises).then( function(response) { // Since an IPV4 interface (e.g. IP address, gateway, or // netmask) edit is a delete then an add and the GUI can't // calculate the interface id (e.g. 5c083707) beforehand and it // is not returned by the REST call, openbmc#3227, reload the // page after an edit, which makes a 2nd REST call. Do this for // all network changes due to the possibility of a set network // failing even though it returned success, openbmc#1641, and to // update dataService and oldInterface to know which data has // changed if the user continues to edit network settings. // TODO: The reload is not ideal. Revisit this. $timeout(function() { loadNetworkInfo(); $scope.loading = false; ngToast.success('Network settings saved'); }, 4000); }, function(error) { $scope.loading = false; ngToast.danger('Network settings could not be saved'); }) } else { $scope.loading = false; } }; function setMACAddress() { return APIUtils .setMACAddress( $scope.selectedInterface, $scope.interface.MACAddress) .then( function(data) {}, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }); } function setDefaultGateway() { return APIUtils.setDefaultGateway($scope.defaultGateway) .then( function(data) {}, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }); } function setHostname() { return APIUtils.setHostname($scope.hostname) .then( function(data) {}, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }); } function setDHCPEnabled() { return APIUtils .setDHCPEnabled( $scope.selectedInterface, $scope.interface.DHCPEnabled) .then( function(data) {}, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }); } function setNameservers() { // Nameservers does not allow an empty array, since we remove all empty // strings above, could have an empty array. TODO: openbmc/openbmc#3240 if ($scope.interface.Nameservers.length == 0) { $scope.interface.Nameservers.push(''); } return APIUtils .setNameservers( $scope.selectedInterface, $scope.interface.Nameservers) .then( function(data) {}, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }); } function removeIPV4s() { return $scope.ipv4sToDelete.map(function(ipv4) { return APIUtils.deleteIPV4($scope.selectedInterface, ipv4.id) .then( function(data) {}, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }) }); } function addIPV4(index) { return APIUtils .addIPV4( $scope.selectedInterface, $scope.interface.ipv4.values[index].Address, $scope.interface.ipv4.values[index].PrefixLength, $scope.interface.ipv4.values[index].Gateway) .then( function(data) {}, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }) } function updateIPV4(index) { // The correct way to edit an IPV4 interface is to remove it and then // add a new one return APIUtils .deleteIPV4( $scope.selectedInterface, $scope.interface.ipv4.values[index].id) .then( function(data) { return APIUtils .addIPV4( $scope.selectedInterface, $scope.interface.ipv4.values[index].Address, $scope.interface.ipv4.values[index].PrefixLength, $scope.interface.ipv4.values[index].Gateway) .then( function(data) {}, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }); }, function(error) { console.log(JSON.stringify(error)); return $q.reject(); }); } $scope.refresh = function() { loadNetworkInfo(); }; function loadNetworkInfo() { APIUtils.getNetworkInfo().then(function(data) { dataService.setNetworkInfo(data); $scope.network = data.formatted_data; $scope.hostname = data.hostname; $scope.defaultGateway = data.defaultgateway; if ($scope.network.interface_ids.length) { // Use the first network interface if the user hasn't chosen one if (!$scope.selectedInterface || !$scope.network.interfaces[$scope.selectedInterface]) { $scope.selectedInterface = $scope.network.interface_ids[0]; } $scope.interface = $scope.network.interfaces[$scope.selectedInterface]; // Copy the interface so we know later if changes were made to the // page $scope.oldInterface = JSON.parse(JSON.stringify($scope.interface)); } // Add id values and update flags to corresponding IPV4 objects for (var i = 0; i < $scope.interface.ipv4.values.length; i++) { $scope.interface.ipv4.values[i].id = $scope.interface.ipv4.ids[i]; $scope.interface.ipv4.values[i].updateAddress = false; $scope.interface.ipv4.values[i].updateGateway = false; $scope.interface.ipv4.values[i].updatePrefix = false; } }); } } ]); })(angular);