2019-03-21 05:46:49 +00:00
|
|
|
import _ from 'lodash-es';
|
2019-11-27 22:36:39 +00:00
|
|
|
import { PorImageRegistryModel } from 'Docker/models/porImageRegistry';
|
2019-03-21 05:46:49 +00:00
|
|
|
|
2020-04-10 21:54:53 +00:00
|
|
|
angular.module('portainer.docker').controller('ImagesController', [
|
|
|
|
'$scope',
|
|
|
|
'$state',
|
2021-03-24 18:27:32 +00:00
|
|
|
'Authentication',
|
2020-04-10 21:54:53 +00:00
|
|
|
'ImageService',
|
|
|
|
'Notifications',
|
|
|
|
'ModalService',
|
|
|
|
'HttpRequestHelper',
|
|
|
|
'FileSaver',
|
|
|
|
'Blob',
|
2021-03-24 18:27:32 +00:00
|
|
|
'endpoint',
|
2021-12-14 07:34:54 +00:00
|
|
|
function ($scope, $state, Authentication, ImageService, Notifications, ModalService, HttpRequestHelper, FileSaver, Blob, endpoint) {
|
2021-03-24 18:27:32 +00:00
|
|
|
$scope.endpoint = endpoint;
|
|
|
|
$scope.isAdmin = Authentication.isAdmin();
|
|
|
|
|
2020-04-10 21:54:53 +00:00
|
|
|
$scope.state = {
|
|
|
|
actionInProgress: false,
|
|
|
|
exportInProgress: false,
|
2021-03-24 18:27:32 +00:00
|
|
|
pullRateValid: false,
|
2020-04-10 21:54:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
$scope.formValues = {
|
|
|
|
RegistryModel: new PorImageRegistryModel(),
|
|
|
|
NodeName: null,
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.pullImage = function () {
|
|
|
|
const registryModel = $scope.formValues.RegistryModel;
|
|
|
|
|
|
|
|
var nodeName = $scope.formValues.NodeName;
|
|
|
|
HttpRequestHelper.setPortainerAgentTargetHeader(nodeName);
|
|
|
|
|
|
|
|
$scope.state.actionInProgress = true;
|
|
|
|
ImageService.pullImage(registryModel, false)
|
2021-07-23 20:51:34 +00:00
|
|
|
.then(function success(data) {
|
|
|
|
var err = data[data.length - 1].errorDetail;
|
|
|
|
if (err) {
|
|
|
|
return Notifications.error('Failure', err, 'Unable to pull image');
|
|
|
|
}
|
2020-04-10 21:54:53 +00:00
|
|
|
Notifications.success('Image successfully pulled', registryModel.Image);
|
|
|
|
$state.reload();
|
|
|
|
})
|
|
|
|
.catch(function error(err) {
|
|
|
|
Notifications.error('Failure', err, 'Unable to pull image');
|
|
|
|
})
|
|
|
|
.finally(function final() {
|
|
|
|
$scope.state.actionInProgress = false;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.confirmRemovalAction = function (selectedItems, force) {
|
|
|
|
ModalService.confirmImageForceRemoval(function (confirmed) {
|
|
|
|
if (!confirmed) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$scope.removeAction(selectedItems, force);
|
2018-07-26 13:09:48 +00:00
|
|
|
});
|
2020-04-10 21:54:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
function isAuthorizedToDownload(selectedItems) {
|
|
|
|
for (var i = 0; i < selectedItems.length; i++) {
|
|
|
|
var image = selectedItems[i];
|
|
|
|
|
|
|
|
var untagged = _.find(image.RepoTags, function (item) {
|
|
|
|
return item.indexOf('<none>') > -1;
|
|
|
|
});
|
2018-07-26 13:09:48 +00:00
|
|
|
|
2020-04-10 21:54:53 +00:00
|
|
|
if (untagged) {
|
|
|
|
Notifications.warning('', 'Cannot download a untagged image');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_.uniqBy(selectedItems, 'NodeName').length > 1) {
|
|
|
|
Notifications.warning('', 'Cannot download images from different nodes at the same time');
|
2019-03-21 05:46:49 +00:00
|
|
|
return false;
|
2018-07-26 13:09:48 +00:00
|
|
|
}
|
|
|
|
|
2020-04-10 21:54:53 +00:00
|
|
|
return true;
|
2018-07-26 13:09:48 +00:00
|
|
|
}
|
|
|
|
|
2020-04-10 21:54:53 +00:00
|
|
|
function exportImages(images) {
|
|
|
|
HttpRequestHelper.setPortainerAgentTargetHeader(images[0].NodeName);
|
|
|
|
$scope.state.exportInProgress = true;
|
|
|
|
ImageService.downloadImages(images)
|
|
|
|
.then(function success(data) {
|
|
|
|
var downloadData = new Blob([data.file], { type: 'application/x-tar' });
|
|
|
|
FileSaver.saveAs(downloadData, 'images.tar');
|
2022-08-10 05:07:35 +00:00
|
|
|
Notifications.success('Success', 'Image(s) successfully downloaded');
|
2020-04-10 21:54:53 +00:00
|
|
|
})
|
|
|
|
.catch(function error(err) {
|
|
|
|
Notifications.error('Failure', err, 'Unable to download image(s)');
|
|
|
|
})
|
|
|
|
.finally(function final() {
|
|
|
|
$scope.state.exportInProgress = false;
|
|
|
|
});
|
2018-07-26 13:09:48 +00:00
|
|
|
}
|
|
|
|
|
2020-04-10 21:54:53 +00:00
|
|
|
$scope.downloadAction = function (selectedItems) {
|
|
|
|
if (!isAuthorizedToDownload(selectedItems)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ModalService.confirmImageExport(function (confirmed) {
|
|
|
|
if (!confirmed) {
|
|
|
|
return;
|
2017-12-06 11:04:02 +00:00
|
|
|
}
|
2020-04-10 21:54:53 +00:00
|
|
|
exportImages(selectedItems);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.removeAction = function (selectedItems, force) {
|
|
|
|
var actionCount = selectedItems.length;
|
|
|
|
angular.forEach(selectedItems, function (image) {
|
|
|
|
HttpRequestHelper.setPortainerAgentTargetHeader(image.NodeName);
|
|
|
|
ImageService.deleteImage(image.Id, force)
|
|
|
|
.then(function success() {
|
|
|
|
Notifications.success('Image successfully removed', image.Id);
|
|
|
|
var index = $scope.images.indexOf(image);
|
|
|
|
$scope.images.splice(index, 1);
|
|
|
|
})
|
|
|
|
.catch(function error(err) {
|
|
|
|
Notifications.error('Failure', err, 'Unable to remove image');
|
|
|
|
})
|
|
|
|
.finally(function final() {
|
|
|
|
--actionCount;
|
|
|
|
if (actionCount === 0) {
|
|
|
|
$state.reload();
|
|
|
|
}
|
|
|
|
});
|
2017-12-06 11:04:02 +00:00
|
|
|
});
|
2020-04-10 21:54:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
$scope.getImages = getImages;
|
|
|
|
function getImages() {
|
|
|
|
ImageService.images(true)
|
|
|
|
.then(function success(data) {
|
|
|
|
$scope.images = data;
|
|
|
|
})
|
|
|
|
.catch(function error(err) {
|
|
|
|
Notifications.error('Failure', err, 'Unable to retrieve images');
|
|
|
|
$scope.images = [];
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-03-24 18:27:32 +00:00
|
|
|
$scope.setPullImageValidity = setPullImageValidity;
|
|
|
|
function setPullImageValidity(validity) {
|
|
|
|
$scope.state.pullRateValid = validity;
|
|
|
|
}
|
|
|
|
|
2020-04-10 21:54:53 +00:00
|
|
|
function initView() {
|
|
|
|
getImages();
|
|
|
|
}
|
|
|
|
|
|
|
|
initView();
|
|
|
|
},
|
|
|
|
]);
|