All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.netflix.fenzo.AssignableVirtualMachine 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.Action1;
import com.netflix.fenzo.plugins.ExclusiveHostConstraint;
import org.apache.mesos.Protos;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.*;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.atomic.AtomicBoolean;

/**
 * This class represents a VM that contains resources that can be assigned to tasks.
 */
class AssignableVirtualMachine implements Comparable{

    /* package */ static final String PseuoHostNamePrefix = "FenzoPsueodHost-";

    private static class PortRange {
        private final VirtualMachineLease.Range range;
        private PortRange(VirtualMachineLease.Range range) {
            this.range = range;
        }
        int size() {
            return range.getEnd()-range.getBeg()+1;
        }
    }

    private static class PortRanges {
        private List ranges = new ArrayList<>();
        private List portRanges = new ArrayList<>();
        private int totalPorts=0;
        private int currUsedPorts=0;

        void addRanges(List ranges) {
            if(ranges!=null) {
                this.ranges.addAll(ranges);
                for(VirtualMachineLease.Range range: ranges) {
                    PortRange pRange = new PortRange(range);
                    portRanges.add(pRange);
                    totalPorts += pRange.size();
                }
            }
        }
        void clear() {
            ranges.clear();
            portRanges.clear();
            currUsedPorts=0;
            totalPorts=0;
        }
        private List getRanges() {
            return ranges;
        }
        boolean hasPorts(int num) {
            return num + currUsedPorts <= totalPorts;
        }
        private int consumeNextPort() {
            int forward=0;
            for(PortRange range: portRanges) {
                if(forward+range.size()>currUsedPorts) {
                    // consume in this range
                    return range.range.getBeg() + (currUsedPorts++ - forward);
                }
                else {
                    forward += range.size();
                }
            }
            throw new IllegalStateException("All ports (" + totalPorts + ") already used up");
        }
    }

    private static class ResAsgmntResult {
        private final List failures;
        private final double fitness;

        public ResAsgmntResult(List failures, double fitness) {
            this.failures = failures;
            this.fitness = fitness;
        }
    }

    private final Map leasesMap;
    private final BlockingQueue workersToUnAssign;
    private final BlockingQueue leasesToExpire;
    private final AtomicBoolean expireAllLeasesNow;
    private final Action1 leaseRejectAction;
    private final long leaseOfferExpirySecs;
    private final String hostname;
    private final Map currTotalScalars = new HashMap<>();
    private final Map currUsedScalars = new HashMap<>();
    private double currTotalCpus=0.0;
    private double currUsedCpus=0.0;
    private double currTotalMemory=0.0;
    private double currUsedMemory=0.0;
    private double currTotalNetworkMbps=0.0;
    private double currUsedNetworkMbps=0.0;
    private double currTotalDisk=0.0;
    private double currUsedDisk=0.0;
    private VirtualMachineLease currTotalLease=null;
    private PortRanges currPortRanges = new PortRanges();
    private volatile Map currAttributesMap = Collections.emptyMap();
    private final Map resourceSets = new HashMap<>();
    // previouslyAssignedTasksMap contains tasks on this VM before current scheduling iteration started. This is
    // available for optimization of scheduling assignments for such things as locality with other similar tasks, etc.
    private final Map previouslyAssignedTasksMap;
    // assignmentResults contains results of assignments on this VM from the current scheduling iteration; they
    // haven't been launched yet
    private final Map assignmentResults;
    private static final Logger logger = LoggerFactory.getLogger(AssignableVirtualMachine.class);
    private final ConcurrentMap leaseIdToHostnameMap;
    private final ConcurrentMap vmIdToHostnameMap;
    private String currVMId =null;
    private final TaskTracker taskTracker;
    private volatile long disabledUntil=0L;
    // This may have to be configurable, but, for now weight the job's soft constraints more than system wide fitness calculators
    private static double softConstraintFitnessWeightPercentage =50.0;
    private static double rSetsFitnessWeightPercentage=15.0;
    private String exclusiveTaskId =null;
    private final boolean singleLeaseMode;
    private boolean firstLeaseAdded=false;
    private final List consumedResourcesToAssign = new ArrayList<>();

    public AssignableVirtualMachine(ConcurrentMap vmIdToHostnameMap,
                                    ConcurrentMap leaseIdToHostnameMap,
                                    String hostname, Action1 leaseRejectAction,
                                    long leaseOfferExpirySecs, TaskTracker taskTracker) {
        this(vmIdToHostnameMap, leaseIdToHostnameMap, hostname, leaseRejectAction, leaseOfferExpirySecs, taskTracker, false);
    }

    public AssignableVirtualMachine(ConcurrentMap vmIdToHostnameMap,
                                    ConcurrentMap leaseIdToHostnameMap,
                                    String hostname, Action1 leaseRejectAction,
                                    long leaseOfferExpirySecs, TaskTracker taskTracker, boolean singleLeaseMode) {
        this.vmIdToHostnameMap = vmIdToHostnameMap;
        this.leaseIdToHostnameMap = leaseIdToHostnameMap;
        this.hostname = hostname;
        this.leaseRejectAction = leaseRejectAction==null?
                new Action1() {
                    @Override
                    public void call(VirtualMachineLease lease) {
                        logger.warn("No lease reject action registered to reject lease id " + lease.getId() +
                                " on host " + lease.hostname());
                    }
                } :
                leaseRejectAction;
        this.leaseOfferExpirySecs = leaseOfferExpirySecs;
        this.taskTracker = taskTracker;
        this.leasesMap = new HashMap<>();
        this.leasesToExpire = new LinkedBlockingQueue<>();
        expireAllLeasesNow = new AtomicBoolean(false);
        this.workersToUnAssign = new LinkedBlockingQueue<>();
        this.previouslyAssignedTasksMap = new HashMap<>();
        this.assignmentResults = new HashMap<>();
        this.singleLeaseMode = singleLeaseMode;
    }

    private void addToAvailableResources(VirtualMachineLease l) {
        if(singleLeaseMode && firstLeaseAdded)
            return; // ToDo should this be illegal state exception?
        firstLeaseAdded = true;
        final Map scalars = l.getScalarValues();
        if(scalars != null && !scalars.isEmpty()) {
            for(Map.Entry entry: scalars.entrySet()) {
                Double currVal = currTotalScalars.get(entry.getKey());
                if(currVal == null)
                    currVal = 0.0;
                currTotalScalars.put(entry.getKey(), currVal + entry.getValue());
            }
        }
        currTotalCpus += l.cpuCores();
        currTotalMemory += l.memoryMB();
        currTotalNetworkMbps += l.networkMbps();
        currTotalDisk += l.diskMB();
        if (l.portRanges() != null)
            currPortRanges.addRanges(l.portRanges());
        if (l.getAttributeMap() != null) {
            // always replace attributes map with the latest
            currAttributesMap = Collections.unmodifiableMap(new HashMap<>(l.getAttributeMap()));
        }
        for(Map.Entry entry: currAttributesMap.entrySet()) {
            switch (entry.getKey()) {
                case "res":
                    String val = entry.getValue().getText().getValue();
                    if(val!=null) {
                        StringTokenizer tokenizer = new StringTokenizer(val, "-");
                        String resName = tokenizer.nextToken();
                        switch (resName) {
                            case PreferentialNamedConsumableResourceSet.attributeName:
                                if(tokenizer.countTokens() == 3) {
                                    String name = tokenizer.nextToken();
                                    String val0Str = tokenizer.nextToken();
                                    String val1Str = tokenizer.nextToken();
                                    if(!resourceSets.containsKey(name)) {
                                        try {
                                            int val0 = Integer.parseInt(val0Str);
                                            int val1 = Integer.parseInt(val1Str);
                                            final PreferentialNamedConsumableResourceSet crs =
                                                    new PreferentialNamedConsumableResourceSet(name, val0, val1);
                                            final Iterator iterator = consumedResourcesToAssign.iterator();
                                            while(iterator.hasNext()) {
                                                TaskRequest request = iterator.next();
                                                crs.assign(request);
                                                iterator.remove();
                                            }
                                            resourceSets.put(name, crs);
                                        }
                                        catch (NumberFormatException e) {
                                            logger.warn(hostname + ": invalid resource spec (" + val + ") in attributes, ignoring: " + e.getMessage());
                                        }
                                    }
                                }
                                else
                                    logger.warn("Invalid res spec (expected 4 tokens with delimiter '-', ignoring: " + val);
                                break;
                            default:
                                logger.warn("Unknown resource in attributes, ignoring: " + val);
                        }
                    }
                    break;
            }
        }
        if(!consumedResourcesToAssign.isEmpty()) {
            throw new IllegalStateException(hostname + ": Some assigned tasks have no resource sets in offers: " +
                    consumedResourcesToAssign);
        }
    }

    void updateCurrTotalLease() {
        currTotalLease = createTotaledLease();
    }

    void resetResources() {
        if(!singleLeaseMode) {
            currTotalCpus=0.0;
            currTotalMemory=0.0;
            currTotalNetworkMbps=0.0;
            currTotalDisk=0.0;
            currPortRanges.clear();
            currTotalScalars.clear();
        }
        currUsedCpus=0.0;
        currUsedMemory=0.0;
        currUsedNetworkMbps=0.0;
        currUsedDisk=0.0;
        currUsedScalars.clear();
        // ToDo: in single offer mode, need to resolve used ports somehow
        // don't clear attribute map
        for(VirtualMachineLease l: leasesMap.values())
            addToAvailableResources(l);
    }

    VirtualMachineLease getCurrTotalLease() {
        return currTotalLease;
    }

    private VirtualMachineLease createTotaledLease() {
        return new VirtualMachineLease() {
            @Override
            public String getId() {
                return "InternalVMLeaseObject";
            }
            @Override
            public long getOfferedTime() {
                return System.currentTimeMillis();
            }
            @Override
            public String hostname() {
                return hostname;
            }
            @Override
            public String getVMID() {
                return "NoVMID-InternalVMLease";
            }
            @Override
            public double cpuCores() {
                return currTotalCpus;
            }
            @Override
            public double memoryMB() {
                return currTotalMemory;
            }
            @Override
            public double networkMbps() {
                return currTotalNetworkMbps;
            }
            @Override
            public double diskMB() {
                return currTotalDisk;
            }
            @Override
            public List portRanges() {
                return Collections.unmodifiableList(currPortRanges.getRanges());
            }
            @Override
            public Protos.Offer getOffer() {
                return null;
            }
            @Override
            public Map getAttributeMap() {
                return currAttributesMap;
            }
            @Override
            public Double getScalarValue(String name) {
                return currTotalScalars.get(name);
            }
            @Override
            public Map getScalarValues() {
                return Collections.unmodifiableMap(currTotalScalars);
            }
        };
    }

    void removeExpiredLeases(boolean all) {
        @SuppressWarnings("MismatchedQueryAndUpdateOfCollection") Set leasesToExpireIds = new HashSet<>();
        leasesToExpire.drainTo(leasesToExpireIds);
        Iterator> iterator = leasesMap.entrySet().iterator();
        boolean expireAll = expireAllLeasesNow.getAndSet(false) || all;
        while(iterator.hasNext()) {
            VirtualMachineLease l = iterator.next().getValue();
            if(expireAll || leasesToExpireIds.contains(l.getId())) {
                leaseIdToHostnameMap.remove(l.getId());
                if(expireAll) {
                    if(logger.isDebugEnabled())
                        logger.debug(hostname + ": expiring lease offer id " + l.getId());
                    leaseRejectAction.call(l);
                }
                iterator.remove();
            }
        }
        if(expireAll && !hasPreviouslyAssignedTasks())
            resourceSets.clear();
    }

    int expireLimitedLeases(AssignableVMs.VMRejectLimiter vmRejectLimiter) {
        if(singleLeaseMode)
            return 0;
        long now = System.currentTimeMillis();
        for (VirtualMachineLease l: leasesMap.values()) {
            if (l.getOfferedTime() < (now - leaseOfferExpirySecs * 1000) && vmRejectLimiter.reject()) {
                for (VirtualMachineLease vml: leasesMap.values()) {
                    leaseIdToHostnameMap.remove(vml.getId());
                    if(logger.isDebugEnabled())
                        logger.debug(getHostname() + ": expiring lease offer id " + l.getId());
                    leaseRejectAction.call(vml);
                }
                int size = leasesMap.values().size();
                leasesMap.clear();
                return size;
            }
        }
        return 0;
    }

    String getCurrVMId() {
        return currVMId;
    }

    boolean addLease(VirtualMachineLease lease) {
        if(singleLeaseMode && firstLeaseAdded)
            return false;
        if(!Objects.equals(currVMId, lease.getVMID())) {
            currVMId = lease.getVMID();
            vmIdToHostnameMap.put(lease.getVMID(), hostname);
        }
        if(System.currentTimeMillis()> entriesIterator = leasesMap.entrySet().iterator();
        while(entriesIterator.hasNext()) {
            Map.Entry entry = entriesIterator.next();
            leaseIdToHostnameMap.remove(entry.getValue().getId());
            leaseRejectAction.call(entry.getValue());
            entriesIterator.remove();
        }
    }

    public void enable() {
        disabledUntil = 0;
    }

    long getDisabledUntil() {
        return disabledUntil;
    }

    boolean isActive() {
        return !leasesMap.isEmpty() ||
                hasPreviouslyAssignedTasks() ||
                !assignmentResults.isEmpty() ||
                !leasesToExpire.isEmpty() ||
                !workersToUnAssign.isEmpty() ||
                System.currentTimeMillis() < disabledUntil;
    }

    boolean isAssignableNow() {
        return !isDisabled() && !leasesMap.isEmpty();
    }

    boolean isDisabled() {
        return System.currentTimeMillis() < disabledUntil;
    }

    void setAssignedTask(TaskRequest request) {
        if(logger.isDebugEnabled())
            logger.debug(getHostname() + ": setting assigned task " + request.getId());
        boolean added = taskTracker.addRunningTask(request, this);
        if(added) {
            assignResourceSets(request);
        }
        else
            logger.error("Unexpected to add duplicate task id=" + request.getId());
        previouslyAssignedTasksMap.put(request.getId(), request);
        setIfExclusive(request);
        if(singleLeaseMode && added) {
            removeResourcesOf(request);
        }
    }

    private void assignResourceSets(TaskRequest request) {
        if(request.getAssignedResources() != null) {
            final List consumedNamedResources =
                    request.getAssignedResources().getConsumedNamedResources();
            if(consumedNamedResources != null && !consumedNamedResources.isEmpty()) {
                for(PreferentialNamedConsumableResourceSet.ConsumeResult cr: consumedNamedResources) {
                    if(resourceSets.get(cr.getAttrName()) == null)
                        consumedResourcesToAssign.add(request); // resource set not available yet
                    else
                        resourceSets.get(cr.getAttrName()).assign(request);
                }
            }
        }
    }

    void expireLease(String leaseId) {
        logger.info("Got request to expire lease on " + hostname);
        leasesToExpire.offer(leaseId);
    }

    void expireAllLeases() {
        expireAllLeasesNow.set(true);
    }

    void markTaskForUnassigning(String taskId) {
        workersToUnAssign.offer(taskId);
    }

    private void setIfExclusive(TaskRequest request) {
        if(request.getHardConstraints()!=null) {
            for(ConstraintEvaluator evaluator: request.getHardConstraints()) {
                if(evaluator instanceof ExclusiveHostConstraint) {
                    exclusiveTaskId = request.getId();
                    return;
                }
            }
        }
    }

    private void clearIfExclusive(String taskId) {
        if(taskId.equals(exclusiveTaskId))
            exclusiveTaskId = null;
    }

    void prepareForScheduling() {
        @SuppressWarnings("MismatchedQueryAndUpdateOfCollection") List tasks = new ArrayList<>();
        workersToUnAssign.drainTo(tasks);
        for(String t: tasks) {
            if(logger.isDebugEnabled())
                logger.debug(getHostname() + ": removing previously assigned task " + t);
            taskTracker.removeRunningTask(t);
            TaskRequest r = previouslyAssignedTasksMap.remove(t);
            if(singleLeaseMode && r!=null)
                addBackResourcesOf(r);
            releaseResourceSets(r);
            clearIfExclusive(t);
        }
        assignmentResults.clear();
    }

    private void releaseResourceSets(TaskRequest r) {
        if(r==null) {
            logger.warn("Can't release resource sets for null task");
            return;
        }
        // unassign resource sets if any
        final Map customNamedResources = r.getCustomNamedResources();
        for (Map.Entry entry : resourceSets.entrySet()) {
            entry.getValue().release(r);
        }
    }

    private void removeResourcesOf(TaskRequest request) {
        currTotalCpus -= request.getCPUs();
        currTotalMemory -= request.getMemory();
        currTotalDisk -= request.getDisk();
        currTotalNetworkMbps -= request.getNetworkMbps();
        final Map scalarRequests = request.getScalarRequests();
        if(scalarRequests != null && !scalarRequests.isEmpty()) {
            for(Map.Entry entry: scalarRequests.entrySet()) {
                Double oldVal = currTotalScalars.get(entry.getKey());
                if(oldVal != null) {
                    double newVal = oldVal - entry.getValue();
                    if(newVal < 0.0) {
                        logger.warn(hostname + ": Scalar resource " + entry.getKey() + " is " + newVal + " after removing " +
                                entry.getValue() + " from task " + request.getId());
                        currTotalScalars.put(entry.getKey(), 0.0);
                    }
                    else
                        currTotalScalars.put(entry.getKey(), newVal);
                }
            }
        }
        // ToDo need to figure out ports as well
    }

    private void addBackResourcesOf(TaskRequest r) {
        currTotalCpus += r.getCPUs();
        currTotalMemory += r.getMemory();
        currTotalNetworkMbps += r.getNetworkMbps();
        currTotalDisk += r.getDisk();
        final Map scalarRequests = r.getScalarRequests();
        if(scalarRequests != null && !scalarRequests.isEmpty()) {
            for(Map.Entry entry: scalarRequests.entrySet()) {
                Double oldVal = currTotalScalars.get(entry.getKey());
                if(oldVal == null)
                    oldVal = 0.0;
                currTotalScalars.put(entry.getKey(), oldVal + entry.getValue());
            }
        }
        // ToDo queueTask back ports
    }

    String getAttrValue(String attrName) {
        if(getCurrTotalLease()==null)
            return null;
        Protos.Attribute attribute = getCurrTotalLease().getAttributeMap().get(attrName);
        if(attribute==null)
            return null;
        return attribute.getText().getValue();
    }

    Map getMaxScalars() {
        Map result = new HashMap<>();
        if (currTotalScalars != null) {
            for (Map.Entry e : currTotalScalars.entrySet()) {
                result.put(e.getKey(), e.getValue());
            }
        }
        if (hasPreviouslyAssignedTasks()) {
            for (TaskRequest t: previouslyAssignedTasksMap.values()) {
                final Map scalarRequests = t.getScalarRequests();
                if (scalarRequests != null && !scalarRequests.isEmpty()) {
                    for (Map.Entry e: scalarRequests.entrySet()) {
                        if (result.get(e.getKey()) == null)
                            result.put(e.getKey(), e.getValue());
                        else
                            result.put(e.getKey(), e.getValue() + result.get(e.getKey()));
                    }
                }
            }
        }
        return result;
    }

    Map getMaxResources() {
        double cpus=0.0;
        double memory=0.0;
        double network=0.0;
        double ports=0.0;
        double disk=0.0;
        for(TaskRequest r: previouslyAssignedTasksMap.values()) {
            cpus += r.getCPUs();
            memory += r.getMemory();
            network += r.getNetworkMbps();
            ports += r.getPorts();
            disk += r.getDisk();
        }
        cpus += getCurrTotalLease().cpuCores();
        memory += getCurrTotalLease().memoryMB();
        network += getCurrTotalLease().networkMbps();
        List ranges = getCurrTotalLease().portRanges();
        for(VirtualMachineLease.Range r: ranges)
            ports += r.getEnd()-r.getBeg();
        disk += getCurrTotalLease().diskMB();
        Map result = new HashMap<>();
        result.put(VMResource.CPU, cpus);
        result.put(VMResource.Memory, memory);
        result.put(VMResource.Network, network);
        result.put(VMResource.Ports, ports);
        result.put(VMResource.Disk, disk);
        return result;
    }

    /**
     * Try assigning resources for a given task.
     * This is the main allocation method to allocate resources from this VM to a given task. This method evaluates
     * hard constraints first. Then, it tries to assign resources. If either of these results in failures, it returns a
     * failure result. If successful, it invokes the fitness calculator to determine the fitness value. Then, it
     * evaluates soft constraints to get its fitness value. The resulting fitness value is reduced as a
     * weighted average of the two fitness values.
     *
     * @param request The task request to assign resources to.
     * @param fitnessCalculator The fitness calculator to use for resource assignment.
     * @return Assignment result.
     */
    TaskAssignmentResult tryRequest(TaskRequest request, VMTaskFitnessCalculator fitnessCalculator) {
        if(logger.isDebugEnabled())
            logger.debug("Host " + getHostname() + " task " + request.getId() + ": #leases=" + leasesMap.size());
        if(leasesMap.isEmpty())
            return null;
        if(exclusiveTaskId!=null) {
            if(logger.isDebugEnabled())
                logger.debug("Host " + getHostname() + ": can't assign task " + request.getId() + ", already have task " +
                        exclusiveTaskId + " assigned with exclusive host constraint");
            ConstraintFailure failure = new ConstraintFailure(ExclusiveHostConstraint.class.getName(),
                    "Already has task " + exclusiveTaskId + " with exclusive host constraint");
            return new TaskAssignmentResult(this, request, false, null, failure, 0.0);
        }
        VirtualMachineCurrentState vmCurrentState = vmCurrentState();
        TaskTrackerState taskTrackerState = taskTrackerState();
        ConstraintFailure failedHardConstraint = findFailedHardConstraints(request, vmCurrentState, taskTrackerState);
        if(failedHardConstraint!=null) {
            if(logger.isDebugEnabled())
                logger.debug("Host " + getHostname() + ": task " + request.getId() + " failed hard constraint: " + failedHardConstraint);
            return new TaskAssignmentResult(this, request, false, null, failedHardConstraint, 0.0);
        }
        final ResAsgmntResult resAsgmntResult = evalAndGetResourceAssignmentFailures(request);
        if(!resAsgmntResult.failures.isEmpty()) {
            if(logger.isDebugEnabled()) {
                StringBuilder b = new StringBuilder();
                for(AssignmentFailure f: resAsgmntResult.failures)
                    b.append(f.toString()).append(" ; ");
                logger.debug(getHostname() + ": task " + request.getId() + " failed assignment: " + b.toString());
            }
            return new TaskAssignmentResult(this, request, false, resAsgmntResult.failures, null, 0.0);
        }
        final double resAsgmntFitness = resAsgmntResult.fitness;
        double fitness = fitnessCalculator.calculateFitness(request, vmCurrentState, taskTrackerState);
        if(fitness == 0.0) {
            if(logger.isDebugEnabled())
                logger.debug(getHostname() + ": task " + request.getId() + " fitness calculator returned 0.0");
            List failures = Collections.singletonList(
                    new AssignmentFailure(VMResource.Fitness, 0.0, 0.0, 0.0, "fitnessCalculator: 0.0"));
            return new TaskAssignmentResult(this, request, false, failures, null, fitness);
        }
        List softConstraints = request.getSoftConstraints();
        // we don't fail on soft constraints
        double softConstraintFitness=1.0;
        if(softConstraints!=null && !softConstraints.isEmpty()) {
            softConstraintFitness = getSoftConstraintsFitness(request, vmCurrentState, taskTrackerState);
        }
        fitness = combineFitnessValues(resAsgmntFitness, fitness, softConstraintFitness);
        return new TaskAssignmentResult(this, request, true, null, null, fitness);
    }

    private double combineFitnessValues(double resAsgmntFitness, double fitness, double softConstraintFitness) {
        return ( resAsgmntFitness * rSetsFitnessWeightPercentage +
                softConstraintFitness * softConstraintFitnessWeightPercentage +
                fitness * (100.0 - rSetsFitnessWeightPercentage - softConstraintFitnessWeightPercentage) )
                / 100.0;
    }

    private double getSoftConstraintsFitness(TaskRequest request, VirtualMachineCurrentState vmCurrentState, TaskTrackerState taskTrackerState) {
        List softConstraints = request.getSoftConstraints();
        int n=0;
        double sum=0.0;
        for(VMTaskFitnessCalculator s: softConstraints) {
            n++;
            sum += s.calculateFitness(request, vmCurrentState, taskTrackerState);
        }
        return sum/n;
    }

    private ResAsgmntResult evalAndGetResourceAssignmentFailures(TaskRequest request) {
        List failures = new ArrayList<>();
        final Map scalarRequests = request.getScalarRequests();
        if(scalarRequests != null && !scalarRequests.isEmpty()) {
            for(Map.Entry entry: scalarRequests.entrySet()) {
                if(entry.getValue() == null)
                    continue;
                Double u = currUsedScalars.get(entry.getKey());
                if(u == null)  u = 0.0;
                Double t = currTotalScalars.get(entry.getKey());
                if(t == null)  t=0.0;
                if(u + entry.getValue() > t) {
                    failures.add(new AssignmentFailure(
                            VMResource.Other, entry.getValue(), u, t, entry.getKey()
                    ));
                }
            }
        }
        if((currUsedCpus+request.getCPUs()) > currTotalCpus) {
            AssignmentFailure failure = new AssignmentFailure(
                    VMResource.CPU, request.getCPUs(), currUsedCpus,
                    currTotalCpus, "");
            //logger.info(hostname+":"+request.getId()+" Insufficient cpus: " + failure.toString());
            failures.add(failure);
        }
        if((currUsedMemory+request.getMemory()) > currTotalMemory) {
            AssignmentFailure failure = new AssignmentFailure(
                    VMResource.Memory, request.getMemory(), currUsedMemory,
                    currTotalMemory, "");
            //logger.info(hostname+":"+request.getId()+" Insufficient memory: " + failure.toString());
            failures.add(failure);
        }
        if((currUsedNetworkMbps+request.getNetworkMbps()) > currTotalNetworkMbps) {
            AssignmentFailure failure = new AssignmentFailure(
                    VMResource.Network, request.getNetworkMbps(), currUsedNetworkMbps, currTotalNetworkMbps, "");
            //logger.info(hostname+":"+request.getId()+" Insufficient network: " + failure.toString());
            failures.add(failure);
        }
        if((currUsedDisk+request.getDisk()) > currTotalDisk) {
            AssignmentFailure failure =
                    new AssignmentFailure(VMResource.Disk, request.getDisk(), currUsedDisk, currTotalDisk, "");
            //logger.info(hostname+":"+request.getId()+" Insufficient disk: " + failure.toString());
            failures.add(failure);
        }
        if(!currPortRanges.hasPorts(request.getPorts())) {
            AssignmentFailure failure = new AssignmentFailure(
                    VMResource.Ports, request.getPorts(), currPortRanges.currUsedPorts,
                    currPortRanges.totalPorts, "");
            //logger.info(hostname+":"+request.getId()+" Insufficient ports: " + failure.toString());
            failures.add(failure);
        }
        double rSetFitness=0.0;
        int numRSets=0;
        final Set requestedNamedResNames = new HashSet<>(request.getCustomNamedResources()==null? Collections.emptySet() :
                request.getCustomNamedResources().keySet());
        if(failures.isEmpty()) {
            // perform resource set checks only if no other assignment failures so far
            for (Map.Entry entry : resourceSets.entrySet()) {
                if (!requestedNamedResNames.isEmpty())
                    requestedNamedResNames.remove(entry.getKey());
                final double fitness = entry.getValue().getFitness(request);
                if (fitness == 0.0) {
                    AssignmentFailure failure = new AssignmentFailure(VMResource.ResourceSet, 0.0, 0.0, 0.0,
                            "ResourceSet " + entry.getValue().getName() + " unavailable"
                    );
                    failures.add(failure);
                } else {
                    rSetFitness += fitness;
                    numRSets++;
                }
            }
            if (!requestedNamedResNames.isEmpty()) {
                // task requested resourceSets that aren't available on this host
                AssignmentFailure failure = new AssignmentFailure(VMResource.ResourceSet, 0.0, 0.0, 0.0,
                        "UnavailableResourceSets: " + requestedNamedResNames
                );
                failures.add(failure);
            } else {
                if (!failures.isEmpty()) {
                    rSetFitness = 0.0;
                } else if (numRSets > 1)
                    rSetFitness /= numRSets;
            }
        }
        return new ResAsgmntResult(failures, rSetFitness);
    }

    private TaskTrackerState taskTrackerState() {
        return new TaskTrackerState() {
            @Override
            public Map getAllRunningTasks() {
                return taskTracker.getAllRunningTasks();
            }

            @Override
            public Map getAllCurrentlyAssignedTasks() {
                return taskTracker.getAllAssignedTasks();
            }
        };
    }

    VirtualMachineCurrentState getVmCurrentState() {
        final List offers = new LinkedList<>();
        for (VirtualMachineLease l: leasesMap.values()) {
            offers.add(l.getOffer());
        }
        return new VirtualMachineCurrentState() {
            @Override
            public String getHostname() {
                return hostname;
            }
            @Override
            public Map getResourceSets() {
                return resourceSets;
            }
            @Override
            public VirtualMachineLease getCurrAvailableResources() {
                return currTotalLease;
            }

            @Override
            public Collection getAllCurrentOffers() {
                System.out.println("****************************** ");
                return offers;
            }

            @Override
            public Collection getTasksCurrentlyAssigned() {
                return Collections.emptyList();
            }
            @Override
            public Collection getRunningTasks() {
                return Collections.unmodifiableCollection(previouslyAssignedTasksMap.values());
            }
            @Override
            public long getDisabledUntil() {
                return disabledUntil;
            }
        };
    }

    private VirtualMachineCurrentState vmCurrentState() {
        final List offers = new LinkedList<>();
        for (VirtualMachineLease l: leasesMap.values()) {
            offers.add(l.getOffer());
        }
        return new VirtualMachineCurrentState() {
            @Override
            public String getHostname() {
                return hostname;
            }
            @Override
            public Map getResourceSets() {
                return resourceSets;
            }
            @Override
            public VirtualMachineLease getCurrAvailableResources() {
                return currTotalLease;
            }

            @Override
            public Collection getAllCurrentOffers() {
                return offers;
            }

            @Override
            public Collection getTasksCurrentlyAssigned() {
                return Collections.unmodifiableCollection(assignmentResults.values());
            }
            @Override
            public Collection getRunningTasks() {
                return Collections.unmodifiableCollection(previouslyAssignedTasksMap.values());
            }
            @Override
            public long getDisabledUntil() {
                return disabledUntil;
            }
        };
    }

    private ConstraintFailure findFailedHardConstraints(TaskRequest request, VirtualMachineCurrentState vmCurrentState, TaskTrackerState taskTrackerState) {
        List hardConstraints = request.getHardConstraints();
        if(hardConstraints==null || hardConstraints.isEmpty())
            return null;
        for(ConstraintEvaluator c: hardConstraints) {
            ConstraintEvaluator.Result r = c.evaluate(request, vmCurrentState, taskTrackerState);
            if(!r.isSuccessful())
                return new ConstraintFailure(c.getName(), r.getFailureReason());
        }
        return null;
    }

    String getHostname() {
        return hostname;
    }

    boolean hasPreviouslyAssignedTasks() {
        return !previouslyAssignedTasksMap.isEmpty();
    }

    /**
     * Assign the given result and update internal counters for used resources. Use this to assign an individual
     * assignment result within a scheduling iteration.
     *
     * @param result The assignment result to assign.
     */
    void assignResult(TaskAssignmentResult result) {
        final Map scalarRequests = result.getRequest().getScalarRequests();
        if(scalarRequests != null && !scalarRequests.isEmpty()) {
            for(Map.Entry entry: scalarRequests.entrySet()) {
                if(entry.getValue() == null)
                    continue;
                Double u = currUsedScalars.get(entry.getKey());
                if(u == null)  u = 0.0;
                currUsedScalars.put(entry.getKey(), u + entry.getValue());
            }
        }
        currUsedCpus += result.getRequest().getCPUs();
        currUsedMemory += result.getRequest().getMemory();
        currUsedNetworkMbps += result.getRequest().getNetworkMbps();
        currUsedDisk += result.getRequest().getDisk();
        for(int p=0; p entry: resourceSets.entrySet()) {
            result.addResourceSet(entry.getValue().consume(result.getRequest()));
        }
        if(!taskTracker.addAssignedTask(result.getRequest(), this))
            logger.error("Unexpected to re-add task to assigned state, id=" + result.getRequest().getId());
        assignmentResults.put(result.getRequest(), result);
    }

    /**
     * Reset the assignment results of current scheduling iteration and return the total assignment result for this VM.
     * Use this at the end of the scheduling iteration. Include all of the assignment results as well as all of the VM
     * leases available in the result.
     *
     * @return Total assignment result including the tasks assigned and VM leases used.
     */
    VMAssignmentResult resetAndGetSuccessfullyAssignedRequests() {
        if(assignmentResults.isEmpty())
            return null;
        Set result = new HashSet<>();
        for(Map.Entry entry: assignmentResults.entrySet())
            if(entry.getValue().isSuccessful())
                result.add(entry.getValue());
        if(result.isEmpty())
            return null;
        VMAssignmentResult vmar = new VMAssignmentResult(hostname, new ArrayList<>(leasesMap.values()), result);
        if(!singleLeaseMode) {
            for(String l: leasesMap.keySet())
                leaseIdToHostnameMap.remove(l);
            leasesMap.clear();
        }
        assignmentResults.clear();
        return vmar;
    }

    // Only makes sense to get called after leases have been consolidated and total resources set in
    // {@Code setAvailableResources()}
    @Override
    public int compareTo(AssignableVirtualMachine o) {
        if(o == null)
            return -1;
        if(o.leasesMap.isEmpty())
            return -1;
        if(leasesMap.isEmpty())
            return 1;
        return Double.compare(o.currTotalCpus, currTotalCpus);
    }

    /**
     * Get resource status, showing used and available amounts. The available amounts are in addition to the amounts used.
     *
     * @return Map with keys containing resources and values containing corresponding usage represented as a two number
     * array, where the first represents the used amounts and the second represents additional available amounts.
     */
    Map getResourceStatus() {
        Map resourceMap = new HashMap<>();
        double cpusUsed=0.0;
        double memUsed=0.0;
        double portsUsed=0.0;
        double networkUsed=0.0;
        for(TaskRequest r: previouslyAssignedTasksMap.values()) {
            cpusUsed += r.getCPUs();
            memUsed += r.getMemory();
            portsUsed += r.getPorts();
            networkUsed += r.getNetworkMbps();
        }
        double cpusAvail=0.0;
        double memAvail=0.0;
        double portsAvail=0;
        double networkAvail=0.0;
        for(VirtualMachineLease l: leasesMap.values()) {
            cpusAvail += l.cpuCores();
            memAvail += l.memoryMB();
            for(VirtualMachineLease.Range range: l.portRanges())
                portsAvail += range.getEnd()-range.getBeg();
            networkAvail += l.networkMbps();
        }
        resourceMap.put(VMResource.CPU, new Double[]{cpusUsed, cpusAvail});
        resourceMap.put(VMResource.Memory, new Double[]{memUsed, memAvail});
        resourceMap.put(VMResource.Ports, new Double[]{portsUsed, portsAvail});
        resourceMap.put(VMResource.Network, new Double[]{networkUsed, networkAvail});
        // put resource sets
        for(PreferentialNamedConsumableResourceSet rSet: resourceSets.values()) {
            final String name = rSet.getName();
            final List usedCounts = rSet.getUsedCounts();
            int used=0;
            for(Double c: usedCounts) {
                if(c>=0)
                    used++;
            }
            resourceMap.put(VMResource.ResourceSet, new Double[]{(double)used, (double)(usedCounts.size()-used)});
        }
        return resourceMap;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy