/* * This file is part of DRBD Management Console by LINBIT HA-Solutions GmbH * written by Rasto Levrinc. * * Copyright (C) 2009-2010, LINBIT HA-Solutions GmbH. * Copyright (C) 2009-2010, Rasto Levrinc * * DRBD Management Console is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as published * by the Free Software Foundation; either version 2, or (at your option) * any later version. * * DRBD Management Console is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with drbd; see the file COPYING. If not, write to * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ package lcmc.vm.ui.resource; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; import java.util.HashMap; import java.util.HashSet; import java.util.List; import java.util.Map; import javax.inject.Inject; import javax.inject.Named; import javax.swing.ImageIcon; import javax.swing.JComponent; import javax.swing.JPanel; import lcmc.common.domain.AccessMode; import lcmc.common.domain.Application; import lcmc.common.ui.treemenu.TreeMenuController; import lcmc.common.ui.utils.SwingUtils; import lcmc.host.domain.Host; import lcmc.common.domain.StringValue; import lcmc.cluster.service.NetworkService; import lcmc.vm.domain.VmsXml; import lcmc.vm.domain.data.InterfaceData; import lcmc.common.domain.Value; import lcmc.common.ui.Browser; import lcmc.cluster.ui.resource.NetInfo; import lcmc.cluster.ui.widget.Widget; import lcmc.common.domain.util.Tools; import org.w3c.dom.Node; /** * This class holds info about Virtual Interfaces. */ @Named public final class InterfaceInfo extends HardwareInfo { /** Parameters. */ private static final String[] PARAMETERS = {InterfaceData.TYPE, InterfaceData.MAC_ADDRESS, InterfaceData.SOURCE_NETWORK, InterfaceData.SOURCE_BRIDGE, InterfaceData.SCRIPT_PATH, InterfaceData.TARGET_DEV, InterfaceData.MODEL_TYPE}; /** Network parameters. */ private static final String[] NETWORK_PARAMETERS = { InterfaceData.TYPE, InterfaceData.MAC_ADDRESS, InterfaceData.SOURCE_NETWORK, InterfaceData.SCRIPT_PATH, InterfaceData.TARGET_DEV, InterfaceData.MODEL_TYPE}; /** Bridge parameters. */ private static final String[] BRIDGE_PARAMETERS = { InterfaceData.TYPE, InterfaceData.MAC_ADDRESS, InterfaceData.SOURCE_BRIDGE, InterfaceData.SCRIPT_PATH, InterfaceData.TARGET_DEV, InterfaceData.MODEL_TYPE}; /** Field type. */ private static final Map<String, Widget.Type> FIELD_TYPES = new HashMap<String, Widget.Type>(); /** Short name. */ private static final Map<String, String> SHORTNAME_MAP = new HashMap<String, String>(); /** Whether the parameter is editable only in advanced mode. */ private static final Collection<String> IS_ENABLED_ONLY_IN_ADVANCED = new HashSet<String>(Arrays.asList(new String[]{ InterfaceData.MAC_ADDRESS, InterfaceData.TARGET_DEV, InterfaceData.MODEL_TYPE})); /** Whether the parameter is required. */ private static final Collection<String> IS_REQUIRED = new HashSet<String>(Arrays.asList(new String[]{ InterfaceData.TYPE })); /** Preferred values. */ private static final Map<String, Value> PREFERRED_MAP = new HashMap<String, Value>(); /** Possible values. */ private static final Map<String, Value[]> POSSIBLE_VALUES = new HashMap<String, Value[]>(); /** Network interface type */ public static final Value TYPE_NETWORK = new StringValue("network"); public static final Value TYPE_BRIDGE = new StringValue("bridge"); static { FIELD_TYPES.put(InterfaceData.TYPE, Widget.Type.RADIOGROUP); SHORTNAME_MAP.put(InterfaceData.TYPE, "Type"); SHORTNAME_MAP.put(InterfaceData.MAC_ADDRESS, "Mac Address"); SHORTNAME_MAP.put(InterfaceData.SOURCE_NETWORK, "Source Network"); SHORTNAME_MAP.put(InterfaceData.SOURCE_BRIDGE, "Source Bridge"); SHORTNAME_MAP.put(InterfaceData.SCRIPT_PATH, "Script Path"); SHORTNAME_MAP.put(InterfaceData.TARGET_DEV, "Target Device"); SHORTNAME_MAP.put(InterfaceData.MODEL_TYPE, "Model Type"); } static { PREFERRED_MAP.put(InterfaceData.SOURCE_NETWORK, new StringValue("default")); POSSIBLE_VALUES.put(InterfaceData.MODEL_TYPE, new Value[]{new StringValue(), new StringValue("default"), new StringValue("e1000"), new StringValue("ne2k_pci"), new StringValue("pcnet"), new StringValue("rtl8139"), new StringValue("virtio")}); POSSIBLE_VALUES.put(InterfaceData.TYPE, new Value[]{TYPE_NETWORK, TYPE_BRIDGE}); POSSIBLE_VALUES.put(InterfaceData.SCRIPT_PATH, new Value[]{new StringValue(), new StringValue("/etc/xen/scripts/vif-bridge")}); } @Inject private SwingUtils swingUtils; @Inject private NetworkService networkService; /** Source network combo box, so that it can be disabled, depending on * type. */ private final Map<String, Widget> sourceNetworkWi = new HashMap<String, Widget>(); /** Source bridge combo box, so that it can be disabled, depending on * type. */ private final Map<String, Widget> sourceBridgeWi = new HashMap<String, Widget>(); /** Table panel. */ private JComponent tablePanel = null; @Inject private TreeMenuController treeMenuController; void init(final String name, final Browser browser, final DomainInfo vmsVirtualDomainInfo) { super.init(name, browser, vmsVirtualDomainInfo); } /** Adds disk table with only this disk to the main panel. */ @Override protected void addHardwareTable(final JPanel mainPanel) { tablePanel = getTablePanel("Interfaces", DomainInfo.INTERFACES_TABLE, getVMSVirtualDomainInfo().getNewInterfaceBtn()); if (getResource().isNew()) { swingUtils.invokeLater(new Runnable() { @Override public void run() { tablePanel.setVisible(false); } }); } mainPanel.add(tablePanel); } /** Returns service icon in the menu. */ @Override public ImageIcon getMenuIcon(final Application.RunMode runMode) { return NetInfo.NET_INTERFACE_ICON; } /** Returns long description of the specified parameter. */ @Override protected String getParamLongDesc(final String param) { return getParamShortDesc(param); } /** Returns short description of the specified parameter. */ @Override protected String getParamShortDesc(final String param) { final String name = SHORTNAME_MAP.get(param); if (name == null) { return param; } return name; } private String generateMacAddress() { String mac; LOOP: while (true) { mac = Tools.generateVMMacAddress(); for (final Host h : getBrowser().getClusterHosts()) { final VmsXml vmsXml = getBrowser().getVmsXml(h); if (vmsXml != null && vmsXml.getUsedMacAddresses().contains(mac)) { continue LOOP; } } break; } return mac; } /** Returns preferred value for specified parameter. */ @Override protected Value getParamPreferred(final String param) { if (InterfaceData.MAC_ADDRESS.equals(param)) { return new StringValue(generateMacAddress()); } return PREFERRED_MAP.get(param); } /** Returns default value for specified parameter. */ @Override protected Value getParamDefault(final String param) { return null; } /** Returns parameters. */ @Override public String[] getParametersFromXML() { return PARAMETERS.clone(); } /** Returns possible choices for drop down lists. */ @Override protected Value[] getParamPossibleChoices(final String param) { if (InterfaceData.SOURCE_NETWORK.equals(param)) { for (final Host h : getVMSVirtualDomainInfo().getDefinedOnHosts()) { final VmsXml vmsXml = getBrowser().getVmsXml(h); if (vmsXml != null) { final List<Value> networks = vmsXml.getNetworks(); networks.add(0, null); return networks.toArray(new Value[networks.size()]); } } } else if (InterfaceData.SOURCE_BRIDGE.equals(param)) { for (final Host host : getVMSVirtualDomainInfo().getDefinedOnHosts()) { final VmsXml vmsXml = getBrowser().getVmsXml(host); if (vmsXml != null) { final List<Value> bridges = new ArrayList<Value>(networkService.getBridges(host)); bridges.add(0, null); return bridges.toArray(new Value[bridges.size()]); } } } return POSSIBLE_VALUES.get(param); } /** Returns section to which the specified parameter belongs. */ @Override protected String getSection(final String param) { return "Interface Options"; } /** Returns true if the specified parameter is required. */ @Override protected boolean isRequired(final String param) { final Value type = getComboBoxValue(InterfaceData.TYPE); if ((InterfaceData.SOURCE_NETWORK.equals(param) && TYPE_NETWORK.equals(type)) || (InterfaceData.SOURCE_BRIDGE.equals(param) && TYPE_BRIDGE.equals(type))) { return true; } return IS_REQUIRED.contains(param); } /** Returns true if the specified parameter is integer. */ @Override protected boolean isInteger(final String param) { return false; } /** Returns true if the specified parameter is label. */ @Override protected boolean isLabel(final String param) { return false; } /** Returns true if the specified parameter is of time type. */ @Override protected boolean isTimeType(final String param) { return false; } /** Returns whether parameter is checkbox. */ @Override protected boolean isCheckBox(final String param) { return false; } /** Returns the type of the parameter. */ @Override protected String getParamType(final String param) { return "undef"; // TODO: } /** Returns the regexp of the parameter. */ @Override protected String getParamRegexp(final String param) { if (InterfaceData.MAC_ADDRESS.equals(param)) { return "^((([0-9a-fA-F]{2}:){5}[0-9a-fA-F]{2})|generate)?$"; } return null; } /** Returns type of the field. */ @Override protected Widget.Type getFieldType(final String param) { return FIELD_TYPES.get(param); } /** Returns device parameters. */ @Override protected Map<String, String> getHWParameters(final boolean allParams) { swingUtils.invokeAndWait(new Runnable() { @Override public void run() { getInfoPanel(); } }); final String[] params = getRealParametersFromXML(); final Map<String, String> parameters = new HashMap<String, String>(); for (final String param : params) { final Value value = getComboBoxValue(param); if (allParams || InterfaceData.SOURCE_NETWORK.equals(param) || InterfaceData.SOURCE_BRIDGE.equals(param) || !Tools.areEqual(getParamSaved(param), value)) { if (Tools.areEqual(getParamDefault(param), value)) { parameters.put(param, null); } else { parameters.put(param, value.getValueForConfig()); } } } setName(getParamSavedForConfig(InterfaceData.MAC_ADDRESS)); return parameters; } /** Modify device xml. */ @Override protected void modifyXML(final VmsXml vmsXml, final Node node, final String domainName, final Map<String, String> params) { if (vmsXml != null) { vmsXml.modifyInterfaceXML(node, domainName, params); } } /** Applies the changes. */ @Override void apply(final Application.RunMode runMode) { if (Application.isTest(runMode)) { return; } swingUtils.invokeAndWait(new Runnable() { @Override public void run() { getApplyButton().setEnabled(false); getRevertButton().setEnabled(false); getInfoPanel(); } }); waitForInfoPanel(); final Map<String, String> parameters = getHWParameters( getResource().isNew()); for (final Host h : getVMSVirtualDomainInfo().getDefinedOnHosts()) { final VmsXml vmsXml = getBrowser().getVmsXml(h); if (vmsXml != null) { parameters.put(InterfaceData.SAVED_MAC_ADDRESS, getName()); final String domainName = getVMSVirtualDomainInfo().getDomainName(); final Node domainNode = vmsXml.getDomainNode(domainName); modifyXML(vmsXml, domainNode, domainName, parameters); final String virshOptions = getVMSVirtualDomainInfo().getVirshOptions(); vmsXml.saveAndDefine(domainNode, domainName, virshOptions); } getResource().setNew(false); } treeMenuController.reloadNode(getNode(), false); getBrowser().periodicalVmsUpdate( getVMSVirtualDomainInfo().getDefinedOnHosts()); swingUtils.invokeLater(new Runnable() { @Override public void run() { tablePanel.setVisible(true); } }); final String[] params = getRealParametersFromXML(); if (Application.isLive(runMode)) { storeComboBoxValues(params); } checkResourceFields(null, params); } /** Returns data for the table. */ @Override protected Object[][] getTableData(final String tableName) { if (DomainInfo.HEADER_TABLE.equals(tableName)) { return getVMSVirtualDomainInfo().getMainTableData(); } else if (DomainInfo.INTERFACES_TABLE.equals(tableName)) { if (getResource().isNew()) { return new Object[][]{}; } return new Object[][]{getVMSVirtualDomainInfo().getInterfaceDataRow( getName(), null, getVMSVirtualDomainInfo().getInterfaces(), true)}; } return new Object[][]{}; } /** Returns whether this parameter is advanced. */ @Override protected boolean isAdvanced(final String param) { return false; } /** Whether the parameter should be enabled. */ @Override protected String isEnabled(final String param) { return null; } /** Whether the parameter should be enabled only in advanced mode. */ @Override protected AccessMode.Mode isEnabledOnlyInAdvancedMode(final String param) { return IS_ENABLED_ONLY_IN_ADVANCED.contains(param) ? AccessMode.ADVANCED : AccessMode.NORMAL; } /** Returns access type of this parameter. */ @Override protected AccessMode.Type getAccessType(final String param) { return AccessMode.ADMIN; } /** Returns true if the value of the parameter is ok. */ @Override protected boolean checkParam(final String param, final Value newValue) { if (InterfaceData.TYPE.equals(param)) { for (final Map.Entry<String, Widget> entry : sourceNetworkWi.entrySet()) { entry.getValue().setVisible(TYPE_NETWORK.equals(newValue)); } for (final Map.Entry<String, Widget> entry : sourceBridgeWi.entrySet()) { entry.getValue().setVisible(TYPE_BRIDGE.equals(newValue)); } checkOneParam(InterfaceData.SOURCE_NETWORK); checkOneParam(InterfaceData.SOURCE_BRIDGE); } return !isRequired(param) || (newValue != null && !newValue.isNothingSelected()); } /** Returns combo box for parameter. */ @Override protected Widget createWidget(final String param, final String prefix, final int width) { final Widget paramWi = super.createWidget(param, prefix, width); if (InterfaceData.TYPE.equals(param)) { paramWi.setAlwaysEditable(false); } else if (InterfaceData.SOURCE_NETWORK.equals(param)) { if (prefix == null) { sourceNetworkWi.put("", paramWi); } else { sourceNetworkWi.put(prefix, paramWi); } paramWi.setAlwaysEditable(false); } else if (InterfaceData.SOURCE_BRIDGE.equals(param)) { if (prefix == null) { sourceBridgeWi.put("", paramWi); } else { sourceBridgeWi.put(prefix, paramWi); } paramWi.setAlwaysEditable(false); } return paramWi; } /** Updates parameters. */ @Override void updateParameters() { final Map<String, InterfaceData> interfaces = getVMSVirtualDomainInfo().getInterfaces(); if (interfaces != null) { final InterfaceData interfaceData = interfaces.get(getName()); if (interfaceData != null) { for (final String param : getParametersFromXML()) { final Value oldValue = getParamSaved(param); Value value = getParamSaved(param); final Widget wi = getWidget(param, null); for (final Host h : getVMSVirtualDomainInfo().getDefinedOnHosts()) { final VmsXml vmsXml = getBrowser().getVmsXml(h); if (vmsXml != null) { final Value savedValue = interfaceData.getValue(param); if (savedValue != null) { value = savedValue; } } } if (!Tools.areEqual(value, oldValue)) { getResource().setValue(param, value); if (wi != null) { /* only if it is not changed by user. */ wi.setValue(value); } } } } } updateTable(DomainInfo.HEADER_TABLE); updateTable(DomainInfo.INTERFACES_TABLE); checkResourceFields(null, getParametersFromXML()); } /** Returns string representation. */ @Override public String toString() { final StringBuilder s = new StringBuilder(30); final Value source; if (TYPE_NETWORK.equals(getParamSaved(InterfaceData.TYPE))) { source = getParamSaved(InterfaceData.SOURCE_NETWORK); } else { source = getParamSaved(InterfaceData.SOURCE_BRIDGE); } if (source == null) { s.append("new interface..."); } else { s.append(source.getValueForConfig()); } final Value saved = getParamSaved(InterfaceData.MAC_ADDRESS); if (saved != null && !saved.isNothingSelected()) { final String stringForGui = saved.getValueForGui(); s.append(" ("); if (stringForGui.length() > 8) { s.append(stringForGui.substring(8)); } else { s.append(stringForGui); } s.append(')'); } return s.toString(); } /** Removes this interface without confirmation dialog. */ @Override protected void removeMyselfNoConfirm(final Application.RunMode runMode) { if (Application.isTest(runMode)) { return; } final String virshOptions = getVMSVirtualDomainInfo().getVirshOptions(); for (final Host h : getVMSVirtualDomainInfo().getDefinedOnHosts()) { final VmsXml vmsXml = getBrowser().getVmsXml(h); if (vmsXml != null) { final Map<String, String> parameters = new HashMap<String, String>(); parameters.put(InterfaceData.SAVED_MAC_ADDRESS, getName()); vmsXml.removeInterfaceXML( getVMSVirtualDomainInfo().getDomainName(), parameters, virshOptions); } } getBrowser().periodicalVmsUpdate( getVMSVirtualDomainInfo().getDefinedOnHosts()); treeMenuController.removeNode(getNode()); } /** * Returns whether this item is removeable (null), or string why it isn't. */ @Override protected String isRemoveable() { return null; } /** Returns real parameters. */ @Override public String[] getRealParametersFromXML() { if (TYPE_NETWORK.equals(getComboBoxValue(InterfaceData.TYPE))) { return NETWORK_PARAMETERS.clone(); } else { return BRIDGE_PARAMETERS.clone(); } } }