com.netflix.fenzo.VMCollection Maven / Gradle / Ivy
/*
* Copyright 2015 Netflix, Inc.
*
* 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.
*/
package com.netflix.fenzo;
import com.netflix.fenzo.functions.Func1;
import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.function.Function;
import java.util.stream.Stream;
class VMCollection {
private static final String defaultGroupName = "DEFAULT";
private final ConcurrentMap> vms;
private final Func1 newVmCreator;
private final String groupingAttrName;
VMCollection(Func1 func1, String groupingAttrName) {
vms = new ConcurrentHashMap<>();
this.newVmCreator = func1;
this.groupingAttrName = groupingAttrName;
}
Collection getAllVMs() {
List result = new LinkedList<>();
vms.values().forEach(m -> m.values().forEach(result::add));
return result;
}
Collection getGroups() {
return Collections.unmodifiableCollection(vms.keySet());
}
/**
* Create n
psuedo VMs for each group by cloning a VM in each group.
* @param groupCounts Map with keys contain group names and values containing number of agents to clone
* @param ruleGetter Getter function for autoscale rules
* @return Collection of psuedo host names added.
*/
Map> clonePseudoVMsForGroups(Map groupCounts, Func1 ruleGetter) {
if (groupCounts == null || groupCounts.isEmpty())
return Collections.emptyMap();
InternalVMCloner vmCloner = new InternalVMCloner();
Map> result = new HashMap<>();
long now = System.currentTimeMillis();
for (String g: groupCounts.keySet()) {
List hostnames = new LinkedList<>();
result.put(g, hostnames);
final ConcurrentMap map = vms.get(g);
if (map != null) {
// NOTE: a shortcoming here is that the attributes of VMs across a group may not be homogeneous.
// By creating one lease object and cloning from it, we pick one combination of the attributes
// and replicate across all the newly created pseudo VMs. It may be possible to capture the
// unique set of attributes across all existing VMs in the group and replicate that mix within
// the new pseudo VMs. However, we will not do that at this time. So, it is possible that some
// task constraints that depend on the variety of such attributes may fail the task placement.
// We will live with that limitation at this time.
VirtualMachineLease lease = vmCloner.getClonedMaxResourcesLease(map.values());
int n = groupCounts.get(g);
final AutoScaleRule rule = ruleGetter.call(g);
if (rule != null) {
int max = rule.getMaxSize();
if (max < Integer.MAX_VALUE && n > (max + map.size()))
n = max - map.size();
}
for (int i = 0; i < n; i++) {
final String hostname = nextHostname(g, i);
try {
addLease(vmCloner.cloneLease(lease, hostname, now));
} catch (Exception e) {
e.printStackTrace();
}
hostnames.add(hostname);
}
}
}
return result;
}
private String nextHostname(String g, int i) {
return AssignableVirtualMachine.PseuoHostNamePrefix + g + "-" + i;
}
/**
* Remove VM of given name from the given group. This is generally unsafe and intended only to be used by whoever
* uses {@link VMCollection#clonePseudoVMsForGroups(Map, Func1)}.
* @param name
* @param group
*/
/* package */ AssignableVirtualMachine unsafeRemoveVm(String name, String group) {
final ConcurrentMap vmsMap = vms.get(group);
if (vmsMap != null) {
return vmsMap.remove(name);
}
return null;
}
Optional getVmByName(String name) {
return vms.values().stream()
.flatMap(
(Function, Stream>) m
-> m.values().stream()
)
.filter(avm -> name.equals(avm.getHostname()))
.findFirst();
}
AssignableVirtualMachine create(String host) {
return create(host, defaultGroupName);
}
AssignableVirtualMachine create(String host, String group) {
vms.putIfAbsent(group, new ConcurrentHashMap<>());
AssignableVirtualMachine prev = null;
if (!defaultGroupName.equals(group)) {
if (vms.get(defaultGroupName) != null)
prev = vms.get(defaultGroupName).remove(host);
}
vms.get(group).putIfAbsent(host, prev == null? newVmCreator.call(host) : prev);
return vms.get(group).get(host);
}
AssignableVirtualMachine getOrCreate(String host) {
final Optional vmByName = getVmByName(host);
if (vmByName.isPresent())
return vmByName.get();
return create(host, defaultGroupName);
}
private AssignableVirtualMachine getOrCreate(String host, String group) {
vms.putIfAbsent(group, new ConcurrentHashMap<>());
final AssignableVirtualMachine avm = vms.get(group).get(host);
if (avm != null)
return avm;
return create(host, group);
}
boolean addLease(VirtualMachineLease l) {
String group = l.getAttributeMap() == null? null :
l.getAttributeMap().get(groupingAttrName) == null?
null :
l.getAttributeMap().get(groupingAttrName).getText().getValue();
if (group == null)
group = defaultGroupName;
final AssignableVirtualMachine avm = getOrCreate(l.hostname(), group);
return avm.addLease(l);
}
public int size() {
final Optional size = vms.values().stream().map(Map::size).reduce((i1, i2) -> i1 + i2);
return size.isPresent()? size.get() : 0;
}
public int size(String group) {
final ConcurrentMap m = vms.get(group);
return m == null? 0 : m.size();
}
public AssignableVirtualMachine remove(AssignableVirtualMachine avm) {
final String group = avm.getAttrValue(groupingAttrName);
AssignableVirtualMachine removed = null;
if (group != null) {
final ConcurrentMap m = vms.get(group);
if (m != null) {
removed = m.remove(avm.getHostname());
}
}
if (removed != null)
return removed;
final ConcurrentMap m = vms.get(defaultGroupName);
if (m != null)
m.remove(avm.getHostname());
return null;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy