blob: d2d99ba984bd9970edb924cc1177681835cf476e [file] [log] [blame]
/**
* 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',
'toastService',
function(
$scope, $window, APIUtils, dataService, $timeout, $route, $q,
toastService) {
$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)) {
toastService.error(
$scope.interface.ipv4.values[i].Address +
' invalid IP parameter');
$scope.loading = false;
return;
}
if (!APIUtils.validIPV4IP(
$scope.interface.ipv4.values[i].Gateway)) {
toastService.error(
$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) {
toastService.error(
$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;
toastService.success('Network settings saved');
}, 4000);
},
function(error) {
$scope.loading = false;
toastService.error('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);