mirror of https://github.com/portainer/portainer
325 lines
9.6 KiB
JavaScript
325 lines
9.6 KiB
JavaScript
// @@OLD_SERVICE_CONTROLLER: this service should be rewritten to use services.
|
|
// See app/components/templates/templatesController.js as a reference.
|
|
angular.module('createContainer', [])
|
|
.controller('CreateContainerController', ['$q', '$scope', '$state', '$stateParams', '$filter', 'Container', 'ContainerHelper', 'Image', 'ImageHelper', 'Volume', 'Network', 'ResourceControlService', 'Authentication', 'Notifications', 'ContainerService', 'ImageService', 'ControllerDataPipeline', 'FormValidator',
|
|
function ($q, $scope, $state, $stateParams, $filter, Container, ContainerHelper, Image, ImageHelper, Volume, Network, ResourceControlService, Authentication, Notifications, ContainerService, ImageService, ControllerDataPipeline, FormValidator) {
|
|
|
|
$scope.formValues = {
|
|
alwaysPull: true,
|
|
Console: 'none',
|
|
Volumes: [],
|
|
Registry: '',
|
|
NetworkContainer: '',
|
|
Labels: [],
|
|
ExtraHosts: [],
|
|
IPv4: '',
|
|
IPv6: ''
|
|
};
|
|
|
|
$scope.state = {
|
|
formValidationError: ''
|
|
};
|
|
|
|
$scope.config = {
|
|
Image: '',
|
|
Env: [],
|
|
Cmd: '',
|
|
ExposedPorts: {},
|
|
HostConfig: {
|
|
RestartPolicy: {
|
|
Name: 'no'
|
|
},
|
|
PortBindings: [],
|
|
PublishAllPorts: false,
|
|
Binds: [],
|
|
NetworkMode: 'bridge',
|
|
Privileged: false,
|
|
ExtraHosts: [],
|
|
Devices:[]
|
|
},
|
|
NetworkingConfig: {
|
|
EndpointsConfig: {}
|
|
},
|
|
Labels: {}
|
|
};
|
|
|
|
$scope.addVolume = function() {
|
|
$scope.formValues.Volumes.push({ name: '', containerPath: '', readOnly: false, type: 'volume' });
|
|
};
|
|
|
|
$scope.removeVolume = function(index) {
|
|
$scope.formValues.Volumes.splice(index, 1);
|
|
};
|
|
|
|
$scope.addEnvironmentVariable = function() {
|
|
$scope.config.Env.push({ name: '', value: ''});
|
|
};
|
|
|
|
$scope.removeEnvironmentVariable = function(index) {
|
|
$scope.config.Env.splice(index, 1);
|
|
};
|
|
|
|
$scope.addPortBinding = function() {
|
|
$scope.config.HostConfig.PortBindings.push({ hostPort: '', containerPort: '', protocol: 'tcp' });
|
|
};
|
|
|
|
$scope.removePortBinding = function(index) {
|
|
$scope.config.HostConfig.PortBindings.splice(index, 1);
|
|
};
|
|
|
|
$scope.addLabel = function() {
|
|
$scope.formValues.Labels.push({ name: '', value: ''});
|
|
};
|
|
|
|
$scope.removeLabel = function(index) {
|
|
$scope.formValues.Labels.splice(index, 1);
|
|
};
|
|
|
|
$scope.addExtraHost = function() {
|
|
$scope.formValues.ExtraHosts.push({ value: '' });
|
|
};
|
|
|
|
$scope.removeExtraHost = function(index) {
|
|
$scope.formValues.ExtraHosts.splice(index, 1);
|
|
};
|
|
|
|
$scope.addDevice = function() {
|
|
$scope.config.HostConfig.Devices.push({ pathOnHost: '', pathInContainer: '' });
|
|
};
|
|
|
|
$scope.removeDevice = function(index) {
|
|
$scope.config.HostConfig.Devices.splice(index, 1);
|
|
};
|
|
|
|
function prepareImageConfig(config) {
|
|
var image = config.Image;
|
|
var registry = $scope.formValues.Registry;
|
|
var imageConfig = ImageHelper.createImageConfigForContainer(image, registry.URL);
|
|
config.Image = imageConfig.fromImage + ':' + imageConfig.tag;
|
|
$scope.imageConfig = imageConfig;
|
|
}
|
|
|
|
function preparePortBindings(config) {
|
|
var bindings = {};
|
|
config.HostConfig.PortBindings.forEach(function (portBinding) {
|
|
if (portBinding.containerPort) {
|
|
var key = portBinding.containerPort + '/' + portBinding.protocol;
|
|
var binding = {};
|
|
if (portBinding.hostPort && portBinding.hostPort.indexOf(':') > -1) {
|
|
var hostAndPort = portBinding.hostPort.split(':');
|
|
binding.HostIp = hostAndPort[0];
|
|
binding.HostPort = hostAndPort[1];
|
|
} else {
|
|
binding.HostPort = portBinding.hostPort;
|
|
}
|
|
bindings[key] = [binding];
|
|
config.ExposedPorts[key] = {};
|
|
}
|
|
});
|
|
config.HostConfig.PortBindings = bindings;
|
|
}
|
|
|
|
function prepareConsole(config) {
|
|
var value = $scope.formValues.Console;
|
|
var openStdin = true;
|
|
var tty = true;
|
|
if (value === 'tty') {
|
|
openStdin = false;
|
|
} else if (value === 'interactive') {
|
|
tty = false;
|
|
} else if (value === 'none') {
|
|
openStdin = false;
|
|
tty = false;
|
|
}
|
|
config.OpenStdin = openStdin;
|
|
config.Tty = tty;
|
|
}
|
|
|
|
function prepareEnvironmentVariables(config) {
|
|
var env = [];
|
|
config.Env.forEach(function (v) {
|
|
if (v.name && v.value) {
|
|
env.push(v.name + '=' + v.value);
|
|
}
|
|
});
|
|
config.Env = env;
|
|
}
|
|
|
|
function prepareVolumes(config) {
|
|
var binds = [];
|
|
var volumes = {};
|
|
|
|
$scope.formValues.Volumes.forEach(function (volume) {
|
|
var name = volume.name;
|
|
var containerPath = volume.containerPath;
|
|
if (name && containerPath) {
|
|
var bind = name + ':' + containerPath;
|
|
volumes[containerPath] = {};
|
|
if (volume.readOnly) {
|
|
bind += ':ro';
|
|
}
|
|
binds.push(bind);
|
|
}
|
|
});
|
|
config.HostConfig.Binds = binds;
|
|
config.Volumes = volumes;
|
|
}
|
|
|
|
function prepareNetworkConfig(config) {
|
|
var mode = config.HostConfig.NetworkMode;
|
|
var container = $scope.formValues.NetworkContainer;
|
|
var containerName = container;
|
|
if (container && typeof container === 'object') {
|
|
containerName = $filter('trimcontainername')(container.Names[0]);
|
|
if ($scope.applicationState.endpoint.mode.provider === 'DOCKER_SWARM') {
|
|
containerName = $filter('swarmcontainername')(container);
|
|
}
|
|
}
|
|
var networkMode = mode;
|
|
if (containerName) {
|
|
networkMode += ':' + containerName;
|
|
}
|
|
config.HostConfig.NetworkMode = networkMode;
|
|
|
|
config.NetworkingConfig.EndpointsConfig[networkMode] = {
|
|
IPAMConfig: {
|
|
IPv4Address: $scope.formValues.IPv4,
|
|
IPv6Address: $scope.formValues.IPv6
|
|
}
|
|
};
|
|
|
|
$scope.formValues.ExtraHosts.forEach(function (v) {
|
|
if (v.value) {
|
|
config.HostConfig.ExtraHosts.push(v.value);
|
|
}
|
|
});
|
|
}
|
|
|
|
function prepareLabels(config) {
|
|
var labels = {};
|
|
$scope.formValues.Labels.forEach(function (label) {
|
|
if (label.name && label.value) {
|
|
labels[label.name] = label.value;
|
|
}
|
|
});
|
|
config.Labels = labels;
|
|
}
|
|
|
|
function prepareDevices(config) {
|
|
var path = [];
|
|
config.HostConfig.Devices.forEach(function (p) {
|
|
if (p.pathOnHost) {
|
|
if(p.pathInContainer === '') {
|
|
p.pathInContainer = p.pathOnHost;
|
|
}
|
|
path.push({PathOnHost:p.pathOnHost,PathInContainer:p.pathInContainer,CgroupPermissions:'rwm'});
|
|
}
|
|
});
|
|
config.HostConfig.Devices = path;
|
|
}
|
|
|
|
function prepareConfiguration() {
|
|
var config = angular.copy($scope.config);
|
|
config.Cmd = ContainerHelper.commandStringToArray(config.Cmd);
|
|
prepareNetworkConfig(config);
|
|
prepareImageConfig(config);
|
|
preparePortBindings(config);
|
|
prepareConsole(config);
|
|
prepareEnvironmentVariables(config);
|
|
prepareVolumes(config);
|
|
prepareLabels(config);
|
|
prepareDevices(config);
|
|
return config;
|
|
}
|
|
|
|
function initView() {
|
|
Volume.query({}, function (d) {
|
|
$scope.availableVolumes = d.Volumes;
|
|
}, function (e) {
|
|
Notifications.error('Failure', e, 'Unable to retrieve volumes');
|
|
});
|
|
|
|
Network.query({}, function (d) {
|
|
var networks = d;
|
|
if ($scope.applicationState.endpoint.mode.provider === 'DOCKER_SWARM' || $scope.applicationState.endpoint.mode.provider === 'DOCKER_SWARM_MODE') {
|
|
networks = d.filter(function (network) {
|
|
if (network.Scope === 'global') {
|
|
return network;
|
|
}
|
|
});
|
|
$scope.globalNetworkCount = networks.length;
|
|
networks.push({Name: 'bridge'});
|
|
networks.push({Name: 'host'});
|
|
networks.push({Name: 'none'});
|
|
}
|
|
networks.push({Name: 'container'});
|
|
$scope.availableNetworks = networks;
|
|
if (!_.find(networks, {'Name': 'bridge'})) {
|
|
$scope.config.HostConfig.NetworkMode = 'nat';
|
|
}
|
|
}, function (e) {
|
|
Notifications.error('Failure', e, 'Unable to retrieve networks');
|
|
});
|
|
|
|
Container.query({}, function (d) {
|
|
var containers = d;
|
|
$scope.runningContainers = containers;
|
|
}, function(e) {
|
|
Notifications.error('Failure', e, 'Unable to retrieve running containers');
|
|
});
|
|
|
|
}
|
|
|
|
function validateForm(accessControlData, isAdmin) {
|
|
$scope.state.formValidationError = '';
|
|
var error = '';
|
|
error = FormValidator.validateAccessControl(accessControlData, isAdmin);
|
|
|
|
if (error) {
|
|
$scope.state.formValidationError = error;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
$scope.create = function () {
|
|
$('#createContainerSpinner').show();
|
|
|
|
var accessControlData = ControllerDataPipeline.getAccessControlFormData();
|
|
var userDetails = Authentication.getUserDetails();
|
|
var isAdmin = userDetails.role === 1 ? true : false;
|
|
|
|
if (!validateForm(accessControlData, isAdmin)) {
|
|
$('#createContainerSpinner').hide();
|
|
return;
|
|
}
|
|
|
|
var config = prepareConfiguration();
|
|
createContainer(config, accessControlData);
|
|
};
|
|
|
|
function createContainer(config, accessControlData) {
|
|
$q.when(!$scope.formValues.alwaysPull || ImageService.pullImage($scope.config.Image, $scope.formValues.Registry, true))
|
|
.finally(function final() {
|
|
ContainerService.createAndStartContainer(config)
|
|
.then(function success(data) {
|
|
var containerIdentifier = data.Id;
|
|
var userId = Authentication.getUserDetails().ID;
|
|
return ResourceControlService.applyResourceControl('container', containerIdentifier, userId, accessControlData, []);
|
|
})
|
|
.then(function success() {
|
|
Notifications.success('Container successfully created');
|
|
$state.go('containers', {}, {reload: true});
|
|
})
|
|
.catch(function error(err) {
|
|
Notifications.error('Failure', err, 'Unable to create container');
|
|
})
|
|
.finally(function final() {
|
|
$('#createContainerSpinner').hide();
|
|
});
|
|
});
|
|
}
|
|
|
|
initView();
|
|
}]);
|