@@ -89,7 +89,7 @@ def test_connection_button_f(f, success, caption = nil)
caption ||= _("Test Connection")
btn_class = success ? 'btn-success' : 'btn-default'
spinner_class = success ? 'spinner-inverse' : nil
spinner_button_f(f, caption, "testConnection(this)",
spinner_button_f(f, caption, "tfm.computeResource.testConnection(this)",
:id => 'test_connection_button',
:spinner_id => 'test_connection_indicator',
:class => btn_class,
@@ -402,7 +402,7 @@ def allocation_text_f(f)
content_tag :button, _(label), :type => 'button', :href => '#',
:name => 'allocation_radio_btn',
:class => (label == active) ? 'btn btn-default active' : 'btn btn-default',
:onclick => "allocation_switcher(this, '#{label}');",
:onclick => "tfm.computeResource.libvirt.allocationSwitcher(this, '#{label}');",
:data => { :toggle => 'button' }
end.join(' ').html_safe
end)
@@ -1,4 +1,3 @@
<%= javascript 'compute_resource' %>
<%= form_for([@set.compute_profile, @set]) do |f| %>
<%= base_errors_for @set %>
@@ -1,5 +1,3 @@
<%= javascript 'compute_resource' %>
<% title _('Compute Profiles') %>
<% title_actions new_link(_('Create Compute Profile')),
@@ -1,4 +1,3 @@
<%= javascript 'compute_resource' %>
<% title(_("Edit Compute profile: %s") % @compute_profile.name) %>
<% title_actions display_link_if_authorized(_("Back"), hash_for_compute_profiles_path, :class => 'btn btn-default') %>
@@ -1,4 +1,3 @@
<%= javascript "compute_resource" %>
<%= form_for @compute_resource, :html => {:data => {:id => @compute_resource.try(:id)}} do |f| %>
<%= base_errors_for @compute_resource %>
<ul class="nav nav-tabs" data-tabs="tabs">
@@ -16,7 +15,7 @@
<%= text_f f, :name %>
<%= f.hidden_field(:provider, :id => '') if f.object.uuid.present? %>
<%= selectable_f f, :provider, list_providers, { :include_blank => _("Choose a provider")},
{:label => _("Provider"), :disabled=> !f.object.new_record?, :'data-url'=> provider_selected_compute_resources_path, :onchange => 'providerSelected(this);'} %>
{:label => _("Provider"), :disabled=> !f.object.new_record?, :'data-url'=> provider_selected_compute_resources_path, :onchange => 'tfm.computeResource.providerSelected(this);'} %>
<%= textarea_f f, :description, :rows => 3 %>
<div id='compute_connection'>
<%= render "compute_resources/form/#{@compute_resource.provider.downcase}", :f => f unless @compute_resource.provider.empty? %>
@@ -3,7 +3,7 @@
<%= password_f f, :password, :keep_value => true, :unset => unset_password? %>
<% datacenters = (f.object.uuid.nil? && controller.action_name != 'test_connection') ? [] : f.object.datacenters rescue []%>
<%= selectable_f(f, :uuid, datacenters, {}, {:label => _('Datacenter'),
:onchange => 'ovirt_datacenterSelected();',
:onchange => 'tfm.computeResource.ovirt.datacenterSelected();',
:help_inline_permanent => load_datacenters_button_f(f, !datacenters.empty?) }) %>
<% quotas = (f.object.uuid.nil? && controller.action_name != 'test_connection') ? [] : f.object.quotas.all rescue []%>
<%= select_f f, :ovirt_quota, quotas, :id, :name, {}, :label => _("Quota ID") %>
@@ -1,4 +1,3 @@
<%= javascript 'compute_resource' %>
<% title @compute_resource.name %>
<% title_actions display_link_if_authorized(_("Associate VMs"),
@@ -12,7 +12,7 @@ images = possible_images(compute_resource, arch, os)
<% if compute_resource.subnets.any? %>
<%= select_f f, :subnet_id, compute_resource.subnets, :subnet_id, :cidr_block,
{:include_blank => _("EC2")},
{:label => _("Subnet"), :label_size => "col-md-2", :onchange => "ec2_vpcSelected(this);"} %>
{:label => _("Subnet"), :label_size => "col-md-2", :onchange => "tfm.computeResource.ec2.vpcSelected(this);"} %>
<% end %>
<% groups, vpc_sg_hash, subnet_vpc_hash = security_groups_selectable(compute_resource, f) %>
@@ -1 +1 @@
<%= text_f f, :size_gb, :class => "col-md-2", :label => _("Size (GB)"), :label_size => "col-md-2", :onchange => 'capacity_edit(this)' %>
<%= text_f f, :size_gb, :class => "col-md-2", :label => _("Size (GB)"), :label_size => "col-md-2", :onchange => 'tfm.computeResource.capacityEdit(this)' %>
@@ -18,7 +18,7 @@
<%= select_f f, :image_id, images, :uuid, :name,{:include_blank => (images.empty? || images.size == 1) ? false : _('Please select an image')},
{ :disabled => images.empty? || (params[:host] && params[:host][:provision_method] == 'build'),
:'data-url' => template_selected_compute_resource_path(compute_resource),
:onchange => 'libvirt_image_selected(this);',
:onchange => 'tfm.computeResource.libvirt.imageSelected(this);',
:help_inline => :indicator,
:help_block => _("Image to use"),
:label => _('Image'), :label_size => "col-md-2"} %>
@@ -5,7 +5,7 @@
<%= selectable_f f, :type, libvirt_networks(compute_resource), {},
{ :label => _('Network type'),
:disabled => !new_vm,
:onchange => 'libvirt_network_selected(this)',
:onchange => 'tfm.computeResource.libvirt.networkSelected(this)',
:class => 'libvirt_network',
:label_size => "col-md-3"
} %>
@@ -1,5 +1,5 @@
<%= selectable_f f, :pool_name, compute_resource.storage_pools.map(&:name), { }, :label => _("Storage pool"), :label_size => "col-md-2" %>
<%= text_f f, :capacity, :label => _("Size (GB)"), :onchange => 'capacity_edit(this)', :label_size => "col-md-2" %>
<%= text_f f, :capacity, :label => _("Size (GB)"), :onchange => 'tfm.computeResource.capacityEdit(this)', :label_size => "col-md-2" %>
<%= allocation_text_f f %>
<%= select_f f, :format_type, %w[RAW QCOW2],:downcase, :to_s, { }, :label => _("Type"), :label_size => "col-md-2" %>
@@ -19,7 +19,7 @@
<%= checkbox_f f, :boot_from_volume, {:label => _("Boot from volume"), :label_size => "col-md-2", :help_inline => _("Create new boot volume from image")}, "true", "false" %>
<%= text_f f, :size_gb, { :label => _("New boot volume size (GB)"), :label_size => "col-md-2", :help_inline => _("Defaults to image size if left blank") } %>
<%= selectable_f f, :scheduler_hint_filter, compute_resource.possible_scheduler_hints, { include_blank: 'None' }, { :label => _("Scheduler hint filter"), :label_size => "col-md-2", :'data-url' => scheduler_hint_selected_hosts_path,
:onchange => 'schedulerHintFilterSelected(this);', :disabled => !new_vm } %>
:onchange => 'tfm.computeResource.openstack.schedulerHintFilterSelected(this);', :disabled => !new_vm } %>
<div id="scheduler_hint_wrapper">
<%= if @host && @host.compute_attributes && @host.compute_attributes[:scheduler_hint_filter].present?
render :partial => "compute_resources_vms/form/scheduler_hint_filters", :locals => { :compute_resource => compute_resource, :host => @host, :new_vm => new_vm }
@@ -1,17 +1,16 @@
<%= javascript_tag("$(document).on('ContentLoad', tfm.numFields.initAll)"); %>
<% javascript 'compute_resource' %>
<%= text_f f, :name, :label => _('Name'), :label_size => "col-md-3" if show_vm_name? %>
<% clusters = compute_resource.clusters %>
<%= select_f f, :cluster, clusters, :id, :name, { },
{ :disabled => !new_vm, :'data-url' => cluster_selected_compute_resource_path(compute_resource),
:onchange => 'ovirt_clusterSelected(this);',
:onchange => 'tfm.computeResource.ovirt.clusterSelected(this);',
:help_inline => :indicator,
:label => _('Cluster'), :label_size => "col-md-2" } %>
<%= f.hidden_field :cluster if !new_vm %>
<%= select_f f, :template, compute_resource.templates, :id, :full_name, {:include_blank => _("Select template")},
{ :disabled => !new_vm, :'data-url' => template_selected_compute_resource_path(compute_resource),
:onchange => 'ovirt_templateSelected(this);',
:onchange => 'tfm.computeResource.ovirt.templateSelected(this);',
:help_inline => :indicator,
:help_block => _("oVirt/RHEV template to use"),
:label => _('Template'), :label_size => "col-md-2" } %>
@@ -34,7 +33,7 @@
<div id='image_selection'>
<%= select_f f, :image_id, images, :uuid, :name,{:include_blank => (images.empty? || images.size == 1) ? false : _('Please select an image')},
{ :disabled => true, :'data-url' => template_selected_compute_resource_path(compute_resource),
:onchange => 'ovirt_templateSelected(this);',
:onchange => 'tfm.computeResource.ovirt.templateSelected(this);',
:help_inline => :indicator,
:help_block => _("Image to use"),
:label => _('Image'), :label_size => "col-md-2"} %>
@@ -9,6 +9,6 @@
<%= checkbox_f f, :preallocate, { :checked => f.object.sparse == 'false', :help_inline => _('Uses thin provisioning if unchecked'), :label => _('Preallocate disk'), :label_size => "col-md-2" } %>
<%= field(f, :bootable, :label => _('Bootable'), :label_size => "col-md-2") do
radio_button_f f, :bootable, {:disabled => !new_vm, :value=>'true', :checked => (f.object.bootable == 'true'), :onclick => 'bootable_radio(this)',
radio_button_f f, :bootable, {:disabled => !new_vm, :value=>'true', :checked => (f.object.bootable == 'true'), :onclick => 'tfm.computeResource.ovirt.bootableRadio(this)',
:text => _('Only one volume can be bootable')}
end %>
@@ -1,4 +1,3 @@
<%= javascript 'compute_resource' %>
<%= javascript_tag("$(document).on('ContentLoad', tfm.numFields.initCounter)"); %>
<%= text_f f, :name, :label => _('Name'), :disabled => !new_vm if show_vm_name? %>
<%= counter_f f, :cpus, :label => _('CPUs'), :label_size => 'col-md-2', :'data-soft-max' => compute_resource.max_cpu_count %>
@@ -12,7 +11,7 @@
end %>
<%= selectable_f f, :cluster, compute_resource.clusters, { :include_blank => _('Please select a cluster') },
:class => "col-md-2", :disabled => !new_vm,
:label => _('Cluster'), :onchange => 'vsphereGetResourcePools(this)',
:label => _('Cluster'), :onchange => 'tfm.computeResource.vmware.getResourcePools(this)',
:help_inline => :indicator,
:data => {:url => resource_pools_compute_resource_path(compute_resource)} %>
<%= vsphere_resource_pools(f, compute_resource, !new_vm) %>
@@ -1,4 +1,3 @@
<%= javascript 'compute_resource' %>
<% title _("New Virtual Machine") %>
<%= form_tag compute_resource_vms_path(@compute_resource) do %>
@@ -1,4 +1,4 @@
<%= javascript 'hosts', 'host_edit', 'host_edit_interfaces', 'class_edit', 'compute_resource', 'charts' %>
<%= javascript 'hosts', 'host_edit', 'host_edit_interfaces', 'class_edit', 'charts' %>
<%= render "hosts/dhcp_lease_errors" if has_dhcp_lease_errors?(@host.errors) %>
<%= render "hosts/conflicts" if (!has_dhcp_lease_errors?(@host.errors) && has_conflicts?(@host.errors)) %>
<%= render "hosts/progress" %>
@@ -9,7 +9,6 @@

# config.assets.precompile += %w()
javascript = %w(compute_resource
compute_resources/openstack/host_edit
compute_resources/libvirt/nic_info
compute_resources/ovirt/nic_info
compute_resources/vmware/nic_info
@@ -9,12 +9,14 @@ require('./bundle_flot');
require('./bundle_multiselect');
require('./bundle_select2');
require('./bundle_datatables');
import compute from './foreman_compute_resource';

window.tfm = Object.assign(
window.tfm || {},
{
tools: require('./foreman_tools'),
users: require('./foreman_users'),
computeResource: compute,
sshKeys: require('./foreman_ssh_keys'),
trends: require('./foreman_trends'),
hostgroups: require('./foreman_hostgroups'),
@@ -0,0 +1,18 @@
import $ from 'jquery';

export function vpcSelected({ value }) {
const sgSelect = $('select.security_group_ids');
const securityGroups = JSON.parse(sgSelect.attr('data-security-groups'));
const subnets = JSON.parse(sgSelect.attr('data-subnets'));
// eslint-disable-next-line camelcase
const vpc = value !== '' ? subnets[value] : { vpc_id: 'ec2', subnet_name: 'ec2' };

sgSelect.empty();

// eslint-disable-next-line camelcase
securityGroups[vpc.vpc_id].forEach(({ group_id, group_name }) => {
// eslint-disable-next-line camelcase
sgSelect.append($('<option />').val(group_id).text(`${group_name} - ${vpc.subnet_name}`));
});
sgSelect.multiSelect('refresh');
}
@@ -0,0 +1,94 @@
import $ from 'jquery';
import { showSpinner } from '../foreman_tools';

export function networkSelected(item) {
const selected = $(item).val();
const bridge = $(item).parentsUntil('.fields').parent().find('#bridge');
const nat = $(item).parentsUntil('.fields').parent().find('#nat');

switch (selected) {
case '':
disableDropdown(bridge);
disableDropdown(nat);
break;
case 'network':
disableDropdown(bridge);
enableDropdown(nat);
break;
case 'bridge':
disableDropdown(nat);
enableDropdown(bridge);
break;
default:
break;
}
return false;
}

function disableDropdown(item) {
item.hide();
item.attr('disabled', true);
}

function enableDropdown(item) {
item.attr('disabled', false);
item.find(':input').attr('disabled', false);
item.show();
}

export function imageSelected(item) {
const template = $(item).val();

if (template) {
const url = $(item).attr('data-url');

showSpinner();
$.ajax({
type: 'post',
url,
data: `template_id=${template}`,
success(result) {
let capacity = $('#storage_volumes').children('.fields').find('[id$=capacity]')[0];

if (parseInt(capacity.value.slice(0, -1), 10) < parseInt(result.capacity, 10)) {
capacity.value = `${result.capacity}G`;
}
$('#storage_volumes').children('.fields').find('[id$=format_type]')[0].value = 'qcow2';
},
complete() {
// eslint-disable-next-line no-undef
reloadOnAjaxComplete(item);
}
});
}
}

export function allocationSwitcher(element, action) {
const previous = $(element).parent().find('.active');

previous.removeClass('active');

const capacity = $(element).closest('.fields').find('[id$=capacity]')[0];
let allocation = $(element).closest('.fields').find('[id$=allocation]')[0];

switch (action) {
case 'None':
$(allocation).attr('readonly', 'readonly');
allocation.value = '0G';
break;
case 'Size':
$(allocation).removeAttr('readonly');
allocation.value = '';
$(allocation).focus();
break;
case 'Full':
$(allocation).attr('readonly', 'readonly');
allocation.value = capacity.value;
break;
default:
break;
}

$(element).button('toggle');
return false;
}
@@ -0,0 +1,34 @@
import $ from 'jquery';
import { showSpinner, hideSpinner } from '../foreman_tools';

export function schedulerHintFilterSelected(item) {
let filter = $(item).val();

if (filter === '') {
$('#scheduler_hint_wrapper').empty();
} else {
let url = $(item).attr('data-url');
// eslint-disable-next-line no-undef
let data = serializeForm().replace('method=patch', 'method=post');

showSpinner();
$.ajax({
type: 'post',
url,
data,
complete() {
hideSpinner();
},
error(jqXHR, status, error) {
$('#scheduler_hint_wrapper').html(
// eslint-disable-next-line no-undef
Jed.sprintf(__('Error loading scheduler hint filters information: %s'), error)
);
$('#compute_resource_tab a').addClass('tab-error');
},
success(result) {
$('#scheduler_hint_wrapper').html(result);
}
});
}
}
@@ -0,0 +1,112 @@
import $ from 'jquery';
import { showSpinner } from '../foreman_tools';
import { testConnection } from '../foreman_compute_resource';

export function templateSelected(item) {
const template = $(item).val();

if (!item.disabled) {
const url = $(item).attr('data-url');

showSpinner();
$.ajax({
type: 'post',
url,
data: `template_id=${template}`,
success(result) {
$('[id$=_memory]').val(result.memory);
$('[id$=_cores]').val(result.cores);
$('#network_interfaces').children('.fields').remove();
$.each(result.interfaces, function () {
addNetworkInterface(this);
});
$('#storage_volumes .children_fields >.fields').remove();
$.each(result.volumes, function () {
addVolume(this);
});
const templateSelector = $('#host_compute_attributes_template');

if (templateSelector.is(':disabled')) {
templateSelector.val(result.id).trigger('change');
}
},
complete() {
// eslint-disable-next-line no-undef
reloadOnAjaxComplete(item);
}
});
}
}

// fill in the template interfaces.
function addNetworkInterface({ name, network }) {
const nestedFields = $('#network_interfaces .add_nested_fields');
// no network interfaces update when the network editing is not allowed by the compute resource

if (nestedFields.length > -1) {
// eslint-disable-next-line no-undef
const newId = add_child_node(nestedFields);

$(`[id$=${newId}_name]`).val(name);
$(`[id$=${newId}_network]`).val(network);
}
}

// fill in the template volumes.
// eslint-disable-next-line camelcase
function addVolume({ size_gb, storage_domain, bootable, id }) {
// eslint-disable-next-line no-undef
const newId = add_child_node($('#storage_volumes .add_nested_fields'));

disableElement($(`[id$=${newId}_size_gb]`).val(size_gb));
disableElement($(`[id$=${newId}_storage_domain]`).val(storage_domain));
disableElement($(`[id$=${newId}_bootable_true]`).attr('checked', bootable));
if (id) {
$(`[id$=${newId}_id]`).val(id);
}
$(`[id$=${newId}_storage_domain]`).next().hide();
}

function disableElement(element) {
element.clone().attr('type', 'hidden').appendTo(element);
element.attr('disabled', 'disabled');
}

export function bootableRadio(item) {
let disabled = $('[id$=_bootable_true]:disabled:checked:visible');

$('[id$=_bootable_true]').prop('checked', false);
if (disabled.length > 0) {
disabled.prop('checked', true);
} else {
$(item).prop('checked', true);
}
}
export function clusterSelected(item) {
const cluster = $(item).val();
const url = $(item).attr('data-url');

showSpinner();
$.ajax({
type: 'post',
url,
data: `cluster_id=${cluster}`,
success(result) {
let networkOptions = $('select[id$=_network]').empty();

$.each(result, function () {
networkOptions.append($('<option />').val(this.id).text(this.name));
});
},
complete() {
// eslint-disable-next-line no-undef
reloadOnAjaxComplete(item);
}
});
}

// used by test connection
export function datacenterSelected(item) {
// eslint-disable-next-line no-undef
testConnection($('#test_connection_button'));
}
@@ -0,0 +1,27 @@
import $ from 'jquery';
import { showSpinner, hideSpinner } from '../foreman_tools';

export function getResourcePools(item) {
// eslint-disable-next-line camelcase
const data = { cluster_id: $(item).val() };
let url = $(item).data('url');

showSpinner();
const selectbox = $('select[id$="resource_pool"]');

selectbox.select2('destroy').empty();
$.ajax({
type: 'get',
url,
data,
complete() {
hideSpinner();
},
success(request) {
request.forEach(({ name }) => {
$('<option>').text(name).val(name).appendTo(selectbox);
});
$(selectbox).select2();
}
});
}
@@ -0,0 +1,117 @@
import $ from 'jquery';
import { activateDatatables } from './foreman_tools';
import { notify } from './foreman_toast_notifications';

export default {
ec2: require('./compute_resource/ec2'),
libvirt: require('./compute_resource/libvirt'),
openstack: require('./compute_resource/openstack'),
ovirt: require('./compute_resource/ovirt'),
vmware: require('./compute_resource/vmware'),
capacityEdit,
providerSelected,
testConnection
};

// Common functions used by one or more Compute Resource

// AJAX load vm listing
$(() => {
$('#vms, #images_list, #key_pairs_list').filter('[data-url]').each(function () {
const url = $(this).attr('data-url');

$(this).load(`${url} table`, function (response, status, xhr) {
if (status === 'error') {
$(this).html(
// eslint-disable-next-line no-undef
Jed.sprintf(__('There was an error listing VMs: %(status)s %(statusText)s'), {
status: xhr.status,
statusText: xhr.statusText
})
);
} else {
activateDatatables();
}
});
});
});

// eslint-disable-next-line max-statements
export function providerSelected(item) {
const computeConnection = $('#compute_connection');
const provider = $(item).val();

if (provider === '') {
computeConnection.hide();
$('[type=submit]').attr('disabled', true);
return false;
}
$('[type=submit]').attr('disabled', false);
const url = $(item).attr('data-url');
const data = `provider=${provider}`;

computeConnection.show();
computeConnection.load(`${url} div#compute_connection`, data, () => {
// eslint-disable-next-line no-undef
password_caps_lock_hint();
});

return false;
}

export function testConnection(item) {
let crId = $('form').data('id');

if (crId === undefined || crId === null) {
crId = '';
}

let password = $('input#compute_resource_password').val();

$('.tab-error').removeClass('tab-error');
$('#test_connection_indicator').show();
$.ajax({
type: 'put',
url: $(item).attr('data-url'),
data: `${$('form').serialize()}&cr_id=${crId}`,
success(result) {
let res = $(`<div>${result}</div>`);

$('#compute_connection').html(res.find('#compute_connection'));
$('#compute_connection').prepend(res.find('.alert'));
if (!$('#compute_resource_provider').prop('disabled')) {
$('#compute_resource_password').prop('disabled', false);
}
if (
$('.alert-danger', result).length === 0 &&
$('#compute_connection .has-error', result).length === 0
) {
notify({ message: `<p>${__('Test connection was successful')}</p>`, type: 'success' });
}
},
error({ statusText }) {
notify({
message: `<p>${__('An error occurred while testing the connection: ')}${statusText}</p>`,
type: 'danger'
});
},
complete(result) {
// we need to restore the password field as it is not sent back from the server.
$('input#compute_resource_password').val(password);
$('#test_connection_indicator').hide();
// eslint-disable-next-line no-undef
reloadOnAjaxComplete('#test_connection_indicator');
}
});
}

export function capacityEdit(element) {
let buttons = $(element).closest('.fields').find('button[name=allocation_radio_btn].btn.active');

if (buttons.length > 0 && $(buttons[0]).text() === 'Full') {
let allocation = $(element).closest('.fields').find('[id$=allocation]')[0];

allocation.value = element.value;
}
return false;
}