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

com.vmware.dcm.backend.ortools.OutputIR Maven / Gradle / Ivy

Go to download

Library for building declarative cluster managers. Please refer to the README at github.com/vmware/declarative-cluster-management/ for instructions on setting up solvers before use.

There is a newer version: 0.15.0
Show newest version
/*
 * Copyright 2018-2020 VMware, Inc. All Rights Reserved.
 *
 * SPDX-License-Identifier: BSD-2
 */

package com.vmware.dcm.backend.ortools;

import com.squareup.javapoet.CodeBlock;

import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.stream.Collectors;

/**
 * A simple IR to represent the output Java program that we are generating
 * for the OrTools solver. The IR represents a tree of "blocks", where each block has
 * a set of variable declarations, headers and a body comprised of more blocks.
 * Headers for the time being are just strings.
 *
 * toString() recursively generates the required code.
 */
class OutputIR {
    private static final String TEMP_VAR_PREFIX = "i";
    private final AtomicInteger intermediateVariableCounter;
    private final AtomicInteger tupleVariableCounter;

    OutputIR() {
        this.intermediateVariableCounter = new AtomicInteger(0);
        this.tupleVariableCounter = new AtomicInteger(0);
    }

    Block newBlock(final String name) {
        return new Block(name);
    }

    ForBlock newForBlock(final String name, final CodeBlock block) {
        return new ForBlock(name, block, "");
    }

    ForBlock newForBlock(final String name, final CodeBlock block, final String size) {
        return new ForBlock(name, block, size);
    }

    ForBlock newForBlock(final String name, final List block) {
        return new ForBlock(name, block, "");
    }

    ForBlock newForBlock(final String name, final List block, final String size) {
        return new ForBlock(name, block, size);
    }

    IfBlock newIfBlock(final String name, final String predicate) {
        return new IfBlock(name, predicate);
    }

    /**
     * Base for every type of expression, identified by a non-unique name
     */
    static class BlockExpr {
        private final String name;

        private BlockExpr(final String name) {
            this.name = name;
        }

        protected String getName() {
            return name;
        }
    }

    /**
     * A generic block of code, that starts with a header of strings, followed by a list of blocks. Each
     * block has a set of variable declarations.
     */
    class Block extends BlockExpr {
        private final Set header;
        private final List body;
        private final Declarations declarations;
        private final List insertionOrder = new ArrayList<>();
        private final String tupleName;

        private Block(final String name) {
            super(name);
            this.body = new ArrayList<>();
            this.header = new LinkedHashSet<>();
            this.declarations = new Declarations(name);
            this.tupleName = "t" + tupleVariableCounter.incrementAndGet();
        }

        boolean addHeader(final CodeBlock blockExpr) {
            final StringBlock b = new StringBlock(blockExpr);
            return header.add(b);
        }

        void addBody(final Block block) {
            body.add(block);
            insertionOrder.add(block);
        }

        void addBody(final CodeBlock block) {
            final StringBlock b = new StringBlock(block);
            insertionOrder.add(b);
        }

        String declare(final String expr) {
            final String varName = declarations.add(expr);
            insertionOrder.add(new DeclarationStatement(varName, expr));
            return varName;
        }

        boolean hasDeclaration(final String expr) {
            return declarations.exists(expr);
        }

        String getDeclaredName(final String expr) {
            return declarations.get(expr);
        }

        public ForBlock getForLoopByName(final String name) {
            final List loops = body.stream()
                    .filter(e ->  e instanceof OutputIR.ForBlock &&  e.getName().equals(name))
                    .collect(Collectors.toList());
            assert loops.size() == 1 && loops.get(0) instanceof ForBlock;
            return (ForBlock) loops.get(0);
        }

        public String getTupleName() {
            return tupleName;
        }

        @Override
        public String toString() {
            final CodeBlock.Builder b = CodeBlock.builder();
            header.forEach(
                e -> b.add(e.toString())
            );
            insertionOrder.forEach(
                e -> b.add(e.toString())
            );
            return b.build().toString();
        }
    }

    /**
     * A block representing a nested for loop.
     */
    class ForBlock extends Block {
        private final List loopExpr;
        private final String size;

        private ForBlock(final String name, final CodeBlock loopExpr, final String size) {
            super(name);
            this.loopExpr = List.of(loopExpr);
            this.size = size;
        }

        private ForBlock(final String name, final List loopExpr, final String size) {
            super(name);
            this.loopExpr = loopExpr;
            this.size = size;
        }

        public String getSize() {
            return size;
        }

        @Override
        public String toString() {
            final CodeBlock.Builder b = CodeBlock.builder();
            loopExpr.forEach(
                    e -> b.beginControlFlow(e.toString())
            );
            final String s = super.toString();
            b.add(s);
            loopExpr.forEach(e -> b.endControlFlow());
            return b.build().toString();
        }
    }

    /**
     * A block representing an If statement.
     */
    class IfBlock extends Block {
        private final String predicate;

        IfBlock(final String name, final String predicate) {
            super(name);
            this.predicate = predicate;
        }

        @Override
        public String toString() {
            final CodeBlock.Builder b = CodeBlock.builder();
            b.beginControlFlow(predicate);
            b.addStatement("continue");
            b.endControlFlow();
            return b.build().toString();
        }
    }

    /**
     * Tracks the set of variable declarations within a block. Within a given block, expressions
     * are reused if possible.
     */
    private class Declarations extends BlockExpr {
        private final Map> declarations = new LinkedHashMap<>();

        private Declarations(final String name) {
            super(name);
        }

        boolean exists(final String expression) {
            return declarations.containsKey(expression);
        }

        String get(final String expression) {
            return declarations.get(expression).get(0);
        }

        String add(final String expression) {
            final String varName = TEMP_VAR_PREFIX + intermediateVariableCounter.getAndIncrement();
            declarations.computeIfAbsent(expression,
                                         (k) -> new ArrayList<>()).add(varName);
            return declarations.get(expression).get(0);
        }

        @Override
        public String toString() {
            return "Declarations{" +
                    "declarations=" + declarations +
                    '}';
        }
    }

    /**
     * Represents statements within a block where an expression is assigned to a variable
     */
    private static class DeclarationStatement extends BlockExpr {
        private final String expr;
        private final String varName;

        private DeclarationStatement(final String varName, final String expr) {
            super("declaration");
            this.varName = varName;
            this.expr = expr;
        }

        @Override
        public String toString() {
            return CodeBlock.builder().addStatement("final var $L = $L", varName, expr).build().toString();
        }
    }

    /**
     * A block of code, already in the form of the required output string. Used for boilerplate within the
     * generated code like comments and some initialization statements.
     */
    private static class StringBlock extends BlockExpr {
        private final CodeBlock codeBlock;

        private StringBlock(final CodeBlock codeBlock) {
            super("string-block");
            this.codeBlock = codeBlock;
        }

        @Override
        public String toString() {
            return codeBlock.toString();
        }

        @Override
        public boolean equals(final Object o) {
            if (this == o) {
                return true;
            }
            if (!(o instanceof StringBlock)) {
                return false;
            }
            final StringBlock that = (StringBlock) o;
            return codeBlock.equals(that.codeBlock);
        }

        @Override
        public int hashCode() {
            return Objects.hash(codeBlock);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy