Merge "JSCS cleanup - launch-instance.js"
This commit is contained in:
commit
ab28be583e
@ -65,7 +65,7 @@ class LaunchImageNG(LaunchImage):
|
||||
|
||||
def __init__(self, attrs=None, **kwargs):
|
||||
if attrs is None:
|
||||
attrs = {"ng-controller": "LaunchInstanceModalCtrl"}
|
||||
attrs = {"ng-controller": "LaunchInstanceModalController"}
|
||||
kwargs['preempt'] = True
|
||||
super(LaunchImage, self).__init__(attrs, **kwargs)
|
||||
|
||||
|
@ -354,7 +354,7 @@ class LaunchLinkNG(LaunchLink):
|
||||
|
||||
def __init__(self,
|
||||
attrs={
|
||||
"ng-controller": "LaunchInstanceModalCtrl",
|
||||
"ng-controller": "LaunchInstanceModalController",
|
||||
"ng-click": "openLaunchInstanceWizard(" +
|
||||
"{successUrl: '/project/instances/'})"
|
||||
},
|
||||
|
@ -29,7 +29,7 @@
|
||||
<div class="launchButtons">
|
||||
{% if launch_instance_allowed %}
|
||||
{% if show_ng_launch %}
|
||||
<a ng-controller="LaunchInstanceModalCtrl" ng-click="openLaunchInstanceWizard({successUrl: '/project/network_topology/', dismissUrl: '/project/network_topology/'})" id="instances__action_launch" class="btn btn-default btn-sm btn-launch ajax-modal {% if instance_quota_exceeded %}disabled{% endif %}"><span class="fa fa-cloud-upload"></span> {% if instance_quota_exceeded %}{% trans "Launch Instance (Quota exceeded)"%}{% else %}{% trans "Launch Instance"%}{% endif %}</a>
|
||||
<a ng-controller="LaunchInstanceModalController" ng-click="openLaunchInstanceWizard({successUrl: '/project/network_topology/', dismissUrl: '/project/network_topology/'})" id="instances__action_launch" class="btn btn-default btn-sm btn-launch ajax-modal {% if instance_quota_exceeded %}disabled{% endif %}"><span class="fa fa-cloud-upload"></span> {% if instance_quota_exceeded %}{% trans "Launch Instance (Quota exceeded)"%}{% else %}{% trans "Launch Instance"%}{% endif %}</a>
|
||||
{% endif %}
|
||||
<a href="{% url 'horizon:project:network_topology:launchinstance' %}" id="instances__action_launch" class="btn btn-default btn-sm btn-launch ajax-modal {% if instance_quota_exceeded %}disabled{% endif %}"><span class="fa fa-cloud-upload"></span> {% if instance_quota_exceeded %}{% trans "Launch Instance (Quota exceeded)"%}{% else %}{% trans "Launch Instance"%}{% endif %}</a>
|
||||
{% endif %}
|
||||
|
@ -31,8 +31,11 @@ ADD_JS_FILES = [
|
||||
'dashboard/workflow/decorator.service.js',
|
||||
'dashboard/workflow/workflow.service.js',
|
||||
'dashboard/cloud-services/cloud-services.js',
|
||||
LAUNCH_INST + 'launch-instance.js',
|
||||
LAUNCH_INST + 'launch-instance.model.js',
|
||||
LAUNCH_INST + 'launch-instance.module.js',
|
||||
LAUNCH_INST + 'launch-instance-workflow.service.js',
|
||||
LAUNCH_INST + 'launch-instance-modal.controller.js',
|
||||
LAUNCH_INST + 'launch-instance-wizard.controller.js',
|
||||
LAUNCH_INST + 'launch-instance-model.js',
|
||||
LAUNCH_INST + 'source/source.controller.js',
|
||||
LAUNCH_INST + 'source/source-help.controller.js',
|
||||
LAUNCH_INST + 'flavor/flavor.controller.js',
|
||||
@ -55,8 +58,11 @@ ADD_JS_SPEC_FILES = [
|
||||
'dashboard/dashboard.module.spec.js',
|
||||
'dashboard/workflow/workflow.module.spec.js',
|
||||
'dashboard/cloud-services/cloud-services.spec.js',
|
||||
LAUNCH_INST + 'launch-instance.spec.js',
|
||||
LAUNCH_INST + 'launch-instance.model.spec.js',
|
||||
LAUNCH_INST + 'launch-instance.module.spec.js',
|
||||
LAUNCH_INST + 'launch-instance-workflow.service.spec.js',
|
||||
LAUNCH_INST + 'launch-instance-modal.controller.spec.js',
|
||||
LAUNCH_INST + 'launch-instance-wizard.controller.spec.js',
|
||||
LAUNCH_INST + 'launch-instance-model.spec.js',
|
||||
LAUNCH_INST + 'source/source.spec.js',
|
||||
LAUNCH_INST + 'flavor/flavor.spec.js',
|
||||
LAUNCH_INST + 'network/network.spec.js',
|
||||
|
@ -91,7 +91,6 @@ module.exports = function (config) {
|
||||
* not significant.
|
||||
*/
|
||||
'**/*.module.js',
|
||||
'**/launch-instance.js',
|
||||
|
||||
/**
|
||||
* Followed by other JavaScript files that defines angular providers
|
||||
|
@ -0,0 +1,57 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
angular
|
||||
.module('hz.dashboard.launch-instance')
|
||||
.controller('LaunchInstanceModalController', LaunchInstanceModalController);
|
||||
|
||||
LaunchInstanceModalController.$inject = [
|
||||
'$scope',
|
||||
'$modal',
|
||||
'$window',
|
||||
'hz.dashboard.launch-instance.modal-spec'
|
||||
];
|
||||
|
||||
function LaunchInstanceModalController($scope, $modal, $window, modalSpec) {
|
||||
$scope.openLaunchInstanceWizard = openLaunchInstanceWizard;
|
||||
|
||||
function openLaunchInstanceWizard(launchContext) {
|
||||
var localSpec = {
|
||||
resolve: {
|
||||
launchContext: function () {
|
||||
return launchContext;
|
||||
}
|
||||
}
|
||||
};
|
||||
angular.extend(localSpec, modalSpec);
|
||||
var launchInstanceModal = $modal.open(localSpec);
|
||||
var handleModalClose = function (redirectPropertyName) {
|
||||
return function () {
|
||||
if (launchContext && launchContext[redirectPropertyName]) {
|
||||
$window.location.href = launchContext[redirectPropertyName];
|
||||
}
|
||||
};
|
||||
};
|
||||
launchInstanceModal.result.then(
|
||||
handleModalClose('successUrl'),
|
||||
handleModalClose('dismissUrl')
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
})();
|
@ -0,0 +1,118 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
describe('LaunchInstanceModalController tests', function() {
|
||||
var ctrl;
|
||||
var modal;
|
||||
var scope;
|
||||
var $window;
|
||||
|
||||
beforeEach(module('hz.dashboard'));
|
||||
beforeEach(module(function($provide) {
|
||||
modal = {
|
||||
open: function() {
|
||||
return {
|
||||
result: {
|
||||
then: angular.noop
|
||||
}
|
||||
};
|
||||
}
|
||||
};
|
||||
$window = { location: { href: '/' } };
|
||||
$provide.value('$modal', modal);
|
||||
$provide.value('$modalSpec', {});
|
||||
$provide.value('$window', $window);
|
||||
}));
|
||||
|
||||
beforeEach(inject(function($controller) {
|
||||
scope = {};
|
||||
ctrl = $controller('LaunchInstanceModalController', { $scope: scope });
|
||||
}));
|
||||
|
||||
it('defines the controller', function() {
|
||||
expect(ctrl).toBeDefined();
|
||||
});
|
||||
|
||||
it('defines openLaunchInstanceWizard', function() {
|
||||
expect(scope.openLaunchInstanceWizard).toBeDefined();
|
||||
});
|
||||
|
||||
describe('openLaunchInstanceWizard function tests', function() {
|
||||
var func;
|
||||
var launchContext;
|
||||
|
||||
beforeEach(function() {
|
||||
func = scope.openLaunchInstanceWizard;
|
||||
launchContext = {};
|
||||
});
|
||||
|
||||
it('calls modal.open', function() {
|
||||
spyOn(modal, 'open').and.returnValue({ result: { then: angular.noop } });
|
||||
func(launchContext);
|
||||
expect(modal.open).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('calls modal.open with expected values', function() {
|
||||
spyOn(modal, 'open').and.returnValue({ result: { then: angular.noop } });
|
||||
launchContext = { info: 'information' };
|
||||
func(launchContext);
|
||||
|
||||
var resolve = modal.open.calls.argsFor(0)[0].resolve;
|
||||
expect(resolve).toBeDefined();
|
||||
expect(resolve.launchContext).toBeDefined();
|
||||
expect(resolve.launchContext()).toEqual({ info: 'information' });
|
||||
});
|
||||
|
||||
it('sets up the correct success and failure paths', function() {
|
||||
var successFunc;
|
||||
var errFunc;
|
||||
|
||||
launchContext = { successUrl: '/good/path', dismissUrl: '/bad/path' };
|
||||
spyOn(modal, 'open').and
|
||||
.returnValue({
|
||||
result: {
|
||||
then: function(x, y) { successFunc = x; errFunc = y; }
|
||||
}
|
||||
});
|
||||
func(launchContext);
|
||||
successFunc('successUrl');
|
||||
expect($window.location.href).toBe('/good/path');
|
||||
errFunc('dismissUrl');
|
||||
expect($window.location.href).toBe('/bad/path');
|
||||
});
|
||||
|
||||
it("doesn't redirect if not configured to", function() {
|
||||
var successFunc;
|
||||
var errFunc;
|
||||
launchContext = {};
|
||||
spyOn(modal, 'open').and
|
||||
.returnValue({
|
||||
result: {
|
||||
then: function(x, y) { successFunc = x; errFunc = y; }
|
||||
}
|
||||
});
|
||||
func(launchContext);
|
||||
successFunc('successUrl');
|
||||
expect($window.location.href).toBe('/');
|
||||
errFunc('dismissUrl');
|
||||
expect($window.location.href).toBe('/');
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
})();
|
@ -0,0 +1,36 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
angular
|
||||
.module('hz.dashboard.launch-instance')
|
||||
.controller('LaunchInstanceWizardController', LaunchInstanceWizardController);
|
||||
|
||||
LaunchInstanceWizardController.$inject = [
|
||||
'$scope',
|
||||
'launchInstanceModel',
|
||||
'hz.dashboard.launch-instance.workflow'
|
||||
];
|
||||
|
||||
function LaunchInstanceWizardController($scope, launchInstanceModel, launchInstanceWorkflow) {
|
||||
$scope.workflow = launchInstanceWorkflow;
|
||||
$scope.model = launchInstanceModel;
|
||||
$scope.model.initialize(true);
|
||||
$scope.submit = $scope.model.createInstance;
|
||||
}
|
||||
|
||||
})();
|
@ -0,0 +1,61 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
describe('LaunchInstanceWizardController tests', function() {
|
||||
var ctrl;
|
||||
var model = {
|
||||
createInstance: function() {
|
||||
return 'created';
|
||||
},
|
||||
initialize: angular.noop
|
||||
};
|
||||
var scope = {};
|
||||
|
||||
beforeEach(module('hz.dashboard'));
|
||||
beforeEach(module(function ($provide) {
|
||||
$provide.value('serviceCatalog', {});
|
||||
$provide.value('launchInstanceModel', model);
|
||||
$provide.value('hz.dashboard.launch-instance.workflow', { thing: true });
|
||||
}));
|
||||
beforeEach(inject(function ($controller) {
|
||||
spyOn(model, 'initialize');
|
||||
var locals = {
|
||||
$scope: scope
|
||||
};
|
||||
ctrl = $controller('LaunchInstanceWizardController', locals);
|
||||
}));
|
||||
|
||||
it('defines the controller', function() {
|
||||
expect(ctrl).toBeDefined();
|
||||
});
|
||||
|
||||
it('calls initialize on the given model', function() {
|
||||
expect(scope.model.initialize).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('sets scope.workflow to the given workflow', function() {
|
||||
expect(scope.workflow).toEqual({ thing: true });
|
||||
});
|
||||
|
||||
it('defines scope.submit', function() {
|
||||
expect(scope.submit).toBeDefined();
|
||||
expect(scope.submit()).toBe('created');
|
||||
});
|
||||
});
|
||||
|
||||
})();
|
@ -0,0 +1,82 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
angular
|
||||
.module('hz.dashboard.launch-instance')
|
||||
.factory('hz.dashboard.launch-instance.workflow', launchInstanceWorkflow);
|
||||
|
||||
launchInstanceWorkflow.$inject = [
|
||||
'dashboardBasePath',
|
||||
'hz.dashboard.workflow.factory'
|
||||
];
|
||||
|
||||
function launchInstanceWorkflow(path, dashboardWorkflow) {
|
||||
return dashboardWorkflow({
|
||||
title: gettext('Launch Instance'),
|
||||
|
||||
steps: [
|
||||
{
|
||||
title: gettext('Select Source'),
|
||||
templateUrl: path + 'launch-instance/source/source.html',
|
||||
helpUrl: path + 'launch-instance/source/source.help.html',
|
||||
formName: 'launchInstanceSourceForm'
|
||||
},
|
||||
{
|
||||
title: gettext('Flavor'),
|
||||
templateUrl: path + 'launch-instance/flavor/flavor.html',
|
||||
helpUrl: path + 'launch-instance/flavor/flavor.help.html',
|
||||
formName: 'launchInstanceFlavorForm'
|
||||
},
|
||||
{
|
||||
title: gettext('Networks'),
|
||||
templateUrl: path + 'launch-instance/network/network.html',
|
||||
helpUrl: path + 'launch-instance/network/network.help.html',
|
||||
formName: 'launchInstanceNetworkForm',
|
||||
requiredServiceTypes: ['network']
|
||||
},
|
||||
{
|
||||
title: gettext('Security Groups'),
|
||||
templateUrl: path + 'launch-instance/security-groups/security-groups.html',
|
||||
helpUrl: path + 'launch-instance/security-groups/security-groups.help.html',
|
||||
formName: 'launchInstanceAccessAndSecurityForm'
|
||||
},
|
||||
{
|
||||
title: gettext('Key Pair'),
|
||||
templateUrl: path + 'launch-instance/keypair/keypair.html',
|
||||
helpUrl: path + 'launch-instance/keypair/keypair.help.html',
|
||||
formName: 'launchInstanceKeypairForm'
|
||||
},
|
||||
{
|
||||
title: gettext('Configuration'),
|
||||
templateUrl: path + 'launch-instance/configuration/configuration.html',
|
||||
helpUrl: path + 'launch-instance/configuration/configuration.help.html',
|
||||
formName: 'launchInstanceConfigurationForm'
|
||||
}
|
||||
],
|
||||
|
||||
btnText: {
|
||||
finish: gettext('Launch Instance')
|
||||
},
|
||||
|
||||
btnIcon: {
|
||||
finish: 'fa fa-cloud-download'
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
})();
|
@ -0,0 +1,70 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function() {
|
||||
'use strict';
|
||||
|
||||
describe('hz.dashboard.launch-instance.workflow tests', function () {
|
||||
var launchInstanceWorkflow;
|
||||
|
||||
beforeEach(module('hz.dashboard'));
|
||||
beforeEach(module(function($provide) {
|
||||
// Need to mock hz.framework.workflow from 'horizon'
|
||||
var workflow = function(spec, decorators) {
|
||||
angular.forEach(decorators, function(decorator) {
|
||||
decorator(spec);
|
||||
});
|
||||
return spec;
|
||||
};
|
||||
$provide.value('horizon.openstack-service-api.serviceCatalog', {});
|
||||
$provide.value('horizon.framework.util.workflow.service', workflow);
|
||||
}));
|
||||
|
||||
beforeEach(inject(function ($injector) {
|
||||
launchInstanceWorkflow = $injector.get('hz.dashboard.launch-instance.workflow');
|
||||
}));
|
||||
|
||||
it('should be defined', function () {
|
||||
expect(launchInstanceWorkflow).toBeDefined();
|
||||
});
|
||||
|
||||
it('should have a title property', function () {
|
||||
expect(launchInstanceWorkflow.title).toBeDefined();
|
||||
});
|
||||
|
||||
it('should have the six steps defined', function () {
|
||||
expect(launchInstanceWorkflow.steps).toBeDefined();
|
||||
expect(launchInstanceWorkflow.steps.length).toBe(6);
|
||||
|
||||
var forms = [
|
||||
'launchInstanceSourceForm',
|
||||
'launchInstanceFlavorForm',
|
||||
'launchInstanceNetworkForm',
|
||||
'launchInstanceAccessAndSecurityForm',
|
||||
'launchInstanceKeypairForm',
|
||||
'launchInstanceConfigurationForm'
|
||||
];
|
||||
|
||||
forms.forEach(function(expectedForm, idx) {
|
||||
expect(launchInstanceWorkflow.steps[idx].formName).toBe(expectedForm);
|
||||
});
|
||||
});
|
||||
|
||||
it('specifies that the network step requires the network service type', function() {
|
||||
expect(launchInstanceWorkflow.steps[2].requiredServiceTypes).toEqual(['network']);
|
||||
});
|
||||
});
|
||||
|
||||
})();
|
@ -1,153 +0,0 @@
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
var module = angular.module('hz.dashboard.launch-instance', [ 'ngSanitize' ]);
|
||||
|
||||
module.factory('launchInstanceWorkflow', [
|
||||
'dashboardBasePath',
|
||||
'hz.dashboard.workflow.factory',
|
||||
|
||||
function (path, dashboardWorkflow) {
|
||||
|
||||
return dashboardWorkflow({
|
||||
title: gettext('Launch Instance'),
|
||||
|
||||
steps: [
|
||||
{
|
||||
title: gettext('Select Source'),
|
||||
templateUrl: path + 'launch-instance/source/source.html',
|
||||
helpUrl: path + 'launch-instance/source/source.help.html',
|
||||
formName: 'launchInstanceSourceForm'
|
||||
},
|
||||
{
|
||||
title: gettext('Flavor'),
|
||||
templateUrl: path + 'launch-instance/flavor/flavor.html',
|
||||
helpUrl: path + 'launch-instance/flavor/flavor.help.html',
|
||||
formName: 'launchInstanceFlavorForm'
|
||||
},
|
||||
{
|
||||
title: gettext('Networks'),
|
||||
templateUrl: path + 'launch-instance/network/network.html',
|
||||
helpUrl: path + 'launch-instance/network/network.help.html',
|
||||
formName: 'launchInstanceNetworkForm',
|
||||
requiredServiceTypes: ['network']
|
||||
},
|
||||
{
|
||||
title: gettext('Security Groups'),
|
||||
templateUrl: path + 'launch-instance/security-groups/security-groups.html',
|
||||
helpUrl: path + 'launch-instance/security-groups/security-groups.help.html',
|
||||
formName: 'launchInstanceAccessAndSecurityForm'
|
||||
},
|
||||
{
|
||||
title: gettext('Key Pair'),
|
||||
templateUrl: path + 'launch-instance/keypair/keypair.html',
|
||||
helpUrl: path + 'launch-instance/keypair/keypair.help.html',
|
||||
formName: 'launchInstanceKeypairForm'
|
||||
},
|
||||
{
|
||||
title: gettext('Configuration'),
|
||||
templateUrl: path + 'launch-instance/configuration/configuration.html',
|
||||
helpUrl: path + 'launch-instance/configuration/configuration.help.html',
|
||||
formName: 'launchInstanceConfigurationForm'
|
||||
}
|
||||
],
|
||||
|
||||
btnText: {
|
||||
finish: gettext('Launch Instance')
|
||||
},
|
||||
|
||||
btnIcon: {
|
||||
finish: 'fa fa-cloud-download'
|
||||
}
|
||||
});
|
||||
}
|
||||
]);
|
||||
|
||||
// Using bootstrap-ui modal widget
|
||||
module.constant('launchInstanceWizardModalSpec', {
|
||||
backdrop: 'static',
|
||||
controller: 'ModalContainerCtrl',
|
||||
template: '<wizard ng-controller="LaunchInstanceWizardCtrl"></wizard>',
|
||||
windowClass: 'modal-dialog-wizard'
|
||||
});
|
||||
|
||||
/**
|
||||
* @name bootSourceTypes
|
||||
* @description Boot source types
|
||||
*/
|
||||
module.constant('bootSourceTypes', {
|
||||
IMAGE: 'image',
|
||||
INSTANCE_SNAPSHOT: 'snapshot',
|
||||
VOLUME: 'volume',
|
||||
VOLUME_SNAPSHOT: 'volume_snapshot'
|
||||
});
|
||||
|
||||
/**
|
||||
* @ngdoc filter
|
||||
* @name diskFormat
|
||||
* @description
|
||||
* Expects object and returns disk_format property value.
|
||||
* Returns empty string if input is null or not an object.
|
||||
* Uniquely required for the source step implementation of transfer tables
|
||||
*/
|
||||
module.filter('diskFormat', function() {
|
||||
return function(input) {
|
||||
if (input === null || !angular.isObject(input) ||
|
||||
!angular.isDefined(input.disk_format) || input.disk_format === null) {
|
||||
return '';
|
||||
} else {
|
||||
return input.disk_format.toUpperCase();
|
||||
}
|
||||
};
|
||||
});
|
||||
|
||||
module.controller('LaunchInstanceWizardCtrl', [
|
||||
'$scope',
|
||||
'launchInstanceModel',
|
||||
'launchInstanceWorkflow',
|
||||
LaunchInstanceWizardCtrl
|
||||
]);
|
||||
|
||||
module.controller('LaunchInstanceModalCtrl', [
|
||||
'$scope',
|
||||
'$modal',
|
||||
'$window',
|
||||
'launchInstanceWizardModalSpec',
|
||||
LaunchInstanceModalCtrl
|
||||
]);
|
||||
|
||||
function LaunchInstanceWizardCtrl($scope, launchInstanceModel, launchInstanceWorkflow) {
|
||||
$scope.workflow = launchInstanceWorkflow;
|
||||
$scope.model = launchInstanceModel;
|
||||
$scope.model.initialize(true);
|
||||
$scope.submit = $scope.model.createInstance;
|
||||
}
|
||||
|
||||
function LaunchInstanceModalCtrl($scope, $modal, $window, modalSpec) {
|
||||
$scope.openLaunchInstanceWizard = function (launchContext) {
|
||||
|
||||
var localSpec = {
|
||||
resolve: {
|
||||
launchContext: function() { return launchContext; }
|
||||
}
|
||||
};
|
||||
|
||||
angular.extend(localSpec, modalSpec);
|
||||
|
||||
var launchInstanceModal = $modal.open(localSpec);
|
||||
|
||||
var handleModalClose = function(redirectPropertyName) {
|
||||
return function() {
|
||||
if (launchContext && launchContext[redirectPropertyName]) {
|
||||
$window.location.href = launchContext[redirectPropertyName];
|
||||
}
|
||||
};
|
||||
};
|
||||
|
||||
launchInstanceModal.result.then(handleModalClose('successUrl'),
|
||||
handleModalClose('dismissUrl'));
|
||||
|
||||
};
|
||||
}
|
||||
|
||||
})();
|
@ -0,0 +1,64 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
angular
|
||||
.module('hz.dashboard.launch-instance', [
|
||||
'ngSanitize'
|
||||
])
|
||||
.constant('hz.dashboard.launch-instance.modal-spec', {
|
||||
backdrop: 'static',
|
||||
controller: 'ModalContainerCtrl',
|
||||
template: '<wizard ng-controller="LaunchInstanceWizardController"></wizard>',
|
||||
windowClass: 'modal-dialog-wizard'
|
||||
})
|
||||
|
||||
/**
|
||||
* @name hz.dashboard.launch-instance.boot-source-types
|
||||
* @description Boot source types
|
||||
*/
|
||||
.constant('hz.dashboard.launch-instance.boot-source-types', {
|
||||
IMAGE: 'image',
|
||||
INSTANCE_SNAPSHOT: 'snapshot',
|
||||
VOLUME: 'volume',
|
||||
VOLUME_SNAPSHOT: 'volume_snapshot'
|
||||
})
|
||||
|
||||
/**
|
||||
* @ngdoc filter
|
||||
* @name diskFormat
|
||||
* @description
|
||||
* Expects object and returns disk_format property value.
|
||||
* Returns empty string if input is null or not an object.
|
||||
* Uniquely required for the source step implementation of transfer tables
|
||||
*/
|
||||
.filter('diskFormat', diskFormat);
|
||||
|
||||
function diskFormat() {
|
||||
return filter;
|
||||
|
||||
function filter(input) {
|
||||
if (input === null || !angular.isObject(input) ||
|
||||
!angular.isDefined(input.disk_format) || input.disk_format === null) {
|
||||
return '';
|
||||
} else {
|
||||
return input.disk_format.toUpperCase();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
})();
|
@ -0,0 +1,40 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
describe('hz.dashboard.launch-instance module', function() {
|
||||
|
||||
beforeEach(module('hz.dashboard'));
|
||||
|
||||
it('should be defined.', function () {
|
||||
expect(angular.module('hz.dashboard.launch-instance')).toBeDefined();
|
||||
});
|
||||
|
||||
describe('hz.dashboard.launch-instance.modal-spec', function () {
|
||||
var launchInstancedModalSpec;
|
||||
|
||||
beforeEach(inject(function ($injector) {
|
||||
launchInstancedModalSpec = $injector.get('hz.dashboard.launch-instance.modal-spec');
|
||||
}));
|
||||
|
||||
it('should be defined', function () {
|
||||
expect(launchInstancedModalSpec).toBeDefined();
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
})();
|
@ -1,233 +0,0 @@
|
||||
/*
|
||||
* (c) Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the 'License');
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an 'AS IS' BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
(function() {
|
||||
'use strict';
|
||||
|
||||
describe('Launch Instance Tests', function() {
|
||||
|
||||
beforeEach(module('hz.dashboard'));
|
||||
|
||||
it('should be defined.', function () {
|
||||
expect(angular.module('hz.dashboard.launch-instance')).toBeDefined();
|
||||
});
|
||||
|
||||
describe('launchInstanceWorkflow', function () {
|
||||
var launchInstanceWorkflow;
|
||||
|
||||
beforeEach(module(function($provide) {
|
||||
// Need to mock hz.framework.workflow from 'horizon'
|
||||
var workflow = function(spec, decorators) {
|
||||
angular.forEach(decorators, function(decorator) {
|
||||
decorator(spec);
|
||||
});
|
||||
return spec;
|
||||
};
|
||||
|
||||
$provide.value('horizon.openstack-service-api.serviceCatalog', {});
|
||||
$provide.value('horizon.framework.util.workflow.service', workflow);
|
||||
}));
|
||||
|
||||
beforeEach(inject(function ($injector) {
|
||||
launchInstanceWorkflow = $injector.get('launchInstanceWorkflow');
|
||||
}));
|
||||
|
||||
it('should be defined', function () {
|
||||
expect(launchInstanceWorkflow).toBeDefined();
|
||||
});
|
||||
|
||||
it('should have a title property', function () {
|
||||
expect(launchInstanceWorkflow.title).toBeDefined();
|
||||
});
|
||||
|
||||
it('should have the six steps defined', function () {
|
||||
expect(launchInstanceWorkflow.steps).toBeDefined();
|
||||
expect(launchInstanceWorkflow.steps.length).toBe(6);
|
||||
|
||||
var forms = [
|
||||
'launchInstanceSourceForm',
|
||||
'launchInstanceFlavorForm',
|
||||
'launchInstanceNetworkForm',
|
||||
'launchInstanceAccessAndSecurityForm',
|
||||
'launchInstanceKeypairForm',
|
||||
'launchInstanceConfigurationForm'
|
||||
];
|
||||
|
||||
forms.forEach(function(expectedForm, idx) {
|
||||
expect(launchInstanceWorkflow.steps[idx].formName).toBe(expectedForm);
|
||||
});
|
||||
});
|
||||
|
||||
it('specifies that the network step requires the network service type', function() {
|
||||
expect(launchInstanceWorkflow.steps[2].requiredServiceTypes).toEqual(['network']);
|
||||
});
|
||||
});
|
||||
|
||||
describe('launchInstanceWizardModalSpec', function () {
|
||||
var launchInstanceWizardModalSpec;
|
||||
|
||||
beforeEach(inject(function ($injector) {
|
||||
launchInstanceWizardModalSpec = $injector.get('launchInstanceWizardModalSpec');
|
||||
}));
|
||||
|
||||
it('should be defined', function () {
|
||||
expect(launchInstanceWizardModalSpec).toBeDefined();
|
||||
});
|
||||
});
|
||||
|
||||
describe('LaunchInstanceWizardCtrl', function() {
|
||||
var ctrl;
|
||||
var model = {
|
||||
createInstance: function() {
|
||||
return 'created';
|
||||
},
|
||||
initialize: angular.noop
|
||||
};
|
||||
|
||||
var scope = {};
|
||||
|
||||
beforeEach(module(function ($provide) {
|
||||
$provide.value('serviceCatalog', {});
|
||||
$provide.value('launchInstanceModel', model);
|
||||
}));
|
||||
|
||||
beforeEach(inject(function ($controller) {
|
||||
spyOn(model, 'initialize');
|
||||
var locals = {
|
||||
$scope: scope,
|
||||
launchInstanceWorkflow: { thing: true }
|
||||
};
|
||||
ctrl = $controller('LaunchInstanceWizardCtrl', locals);
|
||||
}));
|
||||
|
||||
it('defines the controller', function() {
|
||||
expect(ctrl).toBeDefined();
|
||||
});
|
||||
|
||||
it('calls initialize on the given model', function() {
|
||||
expect(scope.model.initialize).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('sets scope.workflow to the given workflow', function() {
|
||||
expect(scope.workflow).toEqual({ thing: true });
|
||||
});
|
||||
|
||||
it('defines scope.submit', function() {
|
||||
expect(scope.submit).toBeDefined();
|
||||
expect(scope.submit()).toBe('created');
|
||||
});
|
||||
|
||||
});
|
||||
|
||||
describe('LaunchInstanceModalCtrl', function() {
|
||||
var ctrl;
|
||||
var modal;
|
||||
var scope;
|
||||
var $window;
|
||||
|
||||
beforeEach(module(function($provide) {
|
||||
modal = {
|
||||
open: function() {
|
||||
return {
|
||||
result: {
|
||||
then: angular.noop
|
||||
}
|
||||
};
|
||||
}
|
||||
};
|
||||
$window = { location: { href: '/' } };
|
||||
|
||||
$provide.value('$modal', modal);
|
||||
$provide.value('$modalSpec', {});
|
||||
$provide.value('$window', $window);
|
||||
}));
|
||||
|
||||
beforeEach(inject(function($controller) {
|
||||
scope = {};
|
||||
ctrl = $controller('LaunchInstanceModalCtrl', { $scope: scope });
|
||||
}));
|
||||
|
||||
it('defines the controller', function() {
|
||||
expect(ctrl).toBeDefined();
|
||||
});
|
||||
|
||||
it('defines openLaunchInstanceWizard', function() {
|
||||
expect(scope.openLaunchInstanceWizard).toBeDefined();
|
||||
});
|
||||
|
||||
describe('openLaunchInstanceWizard function tests', function() {
|
||||
var func;
|
||||
var launchContext;
|
||||
|
||||
beforeEach(function() {
|
||||
func = scope.openLaunchInstanceWizard;
|
||||
launchContext = {};
|
||||
});
|
||||
|
||||
it('calls modal.open', function() {
|
||||
spyOn(modal, 'open').and.returnValue({ result: { then: angular.noop } });
|
||||
func(launchContext);
|
||||
expect(modal.open).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('calls modal.open with expected values', function() {
|
||||
spyOn(modal, 'open').and.returnValue({ result: { then: angular.noop } });
|
||||
launchContext = { info: 'information' };
|
||||
func(launchContext);
|
||||
|
||||
var resolve = modal.open.calls.argsFor(0)[0].resolve;
|
||||
expect(resolve).toBeDefined();
|
||||
expect(resolve.launchContext).toBeDefined();
|
||||
expect(resolve.launchContext()).toEqual({ info: 'information' });
|
||||
});
|
||||
|
||||
it('sets up the correct success and failure paths', function() {
|
||||
var successFunc;
|
||||
var errFunc;
|
||||
|
||||
launchContext = { successUrl: '/good/path', dismissUrl: '/bad/path' };
|
||||
spyOn(modal, 'open').and
|
||||
.returnValue({
|
||||
result: {
|
||||
then: function(x, y) { successFunc = x; errFunc = y; }
|
||||
}
|
||||
});
|
||||
func(launchContext);
|
||||
successFunc('successUrl');
|
||||
expect($window.location.href).toBe('/good/path');
|
||||
errFunc('dismissUrl');
|
||||
expect($window.location.href).toBe('/bad/path');
|
||||
});
|
||||
|
||||
it("doesn't redirect if not configured to", function() {
|
||||
var successFunc;
|
||||
var errFunc;
|
||||
launchContext = {};
|
||||
spyOn(modal, 'open').and
|
||||
.returnValue({
|
||||
result: {
|
||||
then: function(x, y) { successFunc = x; errFunc = y; }
|
||||
}
|
||||
});
|
||||
func(launchContext);
|
||||
successFunc('successUrl');
|
||||
expect($window.location.href).toBe('/');
|
||||
errFunc('dismissUrl');
|
||||
expect($window.location.href).toBe('/');
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
})();
|
@ -32,7 +32,7 @@
|
||||
|
||||
LaunchInstanceSourceController.$inject = [
|
||||
'$scope',
|
||||
'bootSourceTypes',
|
||||
'hz.dashboard.launch-instance.boot-source-types',
|
||||
'bytesFilter',
|
||||
'horizon.framework.widgets.charts.donutChartSettings',
|
||||
'dateFilter',
|
||||
|
@ -25,7 +25,7 @@
|
||||
var scope, ctrl, $browser, deferred;
|
||||
|
||||
beforeEach(module(function($provide) {
|
||||
$provide.value('bootSourceTypes', noop);
|
||||
$provide.value('hz.dashboard.launch-instance.boot-source-types', noop);
|
||||
$provide.value('bytesFilter', noop);
|
||||
$provide.value('horizon.framework.widgets.charts.donutChartSettings', noop);
|
||||
$provide.value('dateFilter', noop);
|
||||
|
Loading…
Reference in New Issue
Block a user