Package org.voltdb.compiler

Source Code of org.voltdb.compiler.AsyncCompilerAgentHelper

/* This file is part of VoltDB.
* Copyright (C) 2008-2014 VoltDB Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with VoltDB.  If not, see <http://www.gnu.org/licenses/>.
*/

package org.voltdb.compiler;

import java.io.IOException;
import java.util.Map.Entry;

import org.voltcore.logging.VoltLogger;
import org.voltcore.utils.Pair;
import org.voltdb.CatalogContext;
import org.voltdb.VoltDB;
import org.voltdb.catalog.Catalog;
import org.voltdb.catalog.CatalogDiffEngine;
import org.voltdb.common.Constants;
import org.voltdb.compiler.ClassMatcher.ClassNameMatchStatus;
import org.voltdb.utils.CatalogUtil;
import org.voltdb.utils.CatalogUtil.CatalogAndIds;
import org.voltdb.utils.Encoder;
import org.voltdb.utils.InMemoryJarfile;

public class AsyncCompilerAgentHelper
{
    private static final VoltLogger compilerLog = new VoltLogger("COMPILER");

    public AsyncCompilerResult prepareApplicationCatalogDiff(CatalogChangeWork work) {
        // create the change result and set up all the boiler plate
        CatalogChangeResult retval = new CatalogChangeResult();
        retval.clientData = work.clientData;
        retval.clientHandle = work.clientHandle;
        retval.connectionId = work.connectionId;
        retval.adminConnection = work.adminConnection;
        retval.hostname = work.hostname;
        retval.invocationType = work.invocationType;
        retval.originalTxnId = work.originalTxnId;
        retval.originalUniqueId = work.originalUniqueId;
        retval.user = work.user;
        retval.tablesThatMustBeEmpty = new String[0]; // ensure non-null

        try {
            // catalog change specific boiler plate
            CatalogContext context = VoltDB.instance().getCatalogContext();
            // Start by assuming we're doing an @UpdateApplicationCatalog.  If-ladder below
            // will complete with newCatalogBytes actually containing the bytes of the
            // catalog to be applied, and deploymentString will contain an actual deployment string,
            // or null if it still needs to be filled in.
            byte[] newCatalogBytes = work.operationBytes;
            String deploymentString = work.operationString;
            if (work.invocationName.equals("@UpdateApplicationCatalog")) {
                // Do the straight-forward thing with the args, filling in nulls as appropriate
                // Grab the current catalog bytes if @UAC had a null catalog
                // (deployment-only update)
                if (newCatalogBytes == null) {
                    try {
                        newCatalogBytes = context.getCatalogJarBytes();
                    }
                    catch (IOException ioe) {
                        retval.errorMsg = "Unexpected exception retrieving internal catalog bytes: " +
                            ioe.getMessage();
                        return retval;
                    }
                }
                // If the deploymentString is null, we'll fill it in with current deployment later
                // Otherwise, deploymentString has the right contents, don't need to touch it
            }
            else if (work.invocationName.equals("@UpdateClasses")) {
                // Need the original catalog bytes, then delete classes, then add
                try {
                    newCatalogBytes = context.getCatalogJarBytes();
                }
                catch (IOException ioe) {
                    retval.errorMsg = "Unexpected exception retrieving internal catalog bytes: " +
                        ioe.getMessage();
                    return retval;
                }
                // provided operationString is really a String with class patterns to delete,
                // provided operationBytes is the jarfile with the upsertable classes
                try {
                    newCatalogBytes = modifyCatalogClasses(newCatalogBytes, work.operationString,
                            work.operationBytes);
                }
                catch (IOException e) {
                    retval.errorMsg = "Unexpected exception @UpdateClasses modifying classes " +
                        "from catalog: " + e.getMessage();
                    return retval;
                }
                // Real deploymentString should be the current deployment, just set it to null
                // here and let it get filled in correctly later.
                deploymentString = null;
            }
            else if (work.invocationName.startsWith("@AdHoc")) {
                // newCatalogBytes and deploymentString should be null.
                // work.adhocDDLStmts should be applied to the current catalog
                try {
                    newCatalogBytes = addDDLToCatalog(context.getCatalogJarBytes(),
                            work.adhocDDLStmts);
                }
                catch (IOException ioe) {
                    retval.errorMsg = "Unexpected exception applying DDL statements to " +
                        " original catalog: " + ioe.getMessage();
                    return retval;
                }
                if (newCatalogBytes == null) {
                    // Shouldn't ever get here
                    retval.errorMsg =
                        "Unexpected failure in applying DDL statements to original catalog";
                    return retval;
                }
                // Real deploymentString should be the current deployment, just set it to null
                // here and let it get filled in correctly later.
                deploymentString = null;
            }
            else {
                retval.errorMsg = "Unexpected work in the AsyncCompilerAgentHelper: " +
                    work.invocationName;
                return retval;
            }
            retval.catalogBytes = newCatalogBytes;
            retval.catalogHash = CatalogUtil.makeCatalogOrDeploymentHash(newCatalogBytes);

            // get the diff between catalogs
            // try to get the new catalog from the params
            Pair<InMemoryJarfile, String> loadResults = null;
            try {
                loadResults = CatalogUtil.loadAndUpgradeCatalogFromJar(newCatalogBytes);
            }
            catch (IOException ioe) {
                // Preserve a nicer message from the jarfile loading rather than
                // falling through to the ZOMG message in the big catch
                retval.errorMsg = ioe.getMessage();
                return retval;
            }
            String newCatalogCommands =
                CatalogUtil.getSerializedCatalogStringFromJar(loadResults.getFirst());
            retval.upgradedFromVersion = loadResults.getSecond();
            if (newCatalogCommands == null) {
                retval.errorMsg = "Unable to read from catalog bytes";
                return retval;
            }
            Catalog newCatalog = new Catalog();
            newCatalog.execute(newCatalogCommands);

            // Retrieve the original deployment string, if necessary
            if (deploymentString == null) {
                // Go get the deployment string from ZK.  Hope it's there and up-to-date.  Yeehaw!
                CatalogAndIds catalogStuff =
                    CatalogUtil.getCatalogFromZK(VoltDB.instance().getHostMessenger().getZK());
                byte[] deploymentBytes = catalogStuff.deploymentBytes;
                if (deploymentBytes != null) {
                    deploymentString = new String(deploymentBytes, "UTF-8");
                }
                if (deploymentBytes == null || deploymentString == null) {
                    retval.errorMsg = "No deployment file provided and unable to recover previous " +
                        "deployment settings.";
                    return retval;
                }
            }

            long result = CatalogUtil.compileDeploymentString(newCatalog, deploymentString, false, false);
            if (result < 0) {
                retval.errorMsg = "Unable to update deployment configuration; see the server logs for more details.";
                return retval;
            }

            retval.deploymentString = deploymentString;
            retval.deploymentHash =
                CatalogUtil.makeCatalogOrDeploymentHash(deploymentString.getBytes("UTF-8"));

            // store the version of the catalog the diffs were created against.
            // verified when / if the update procedure runs in order to verify
            // catalogs only move forward
            retval.expectedCatalogVersion = context.catalogVersion;

            // compute the diff in StringBuilder
            CatalogDiffEngine diff = new CatalogDiffEngine(context.catalog, newCatalog);
            if (!diff.supported()) {
                retval.errorMsg = "The requested catalog change(s) are not supported:\n" + diff.errors();
                return retval;
            }

            // since diff commands can be stupidly big, compress them here
            retval.encodedDiffCommands = Encoder.compressAndBase64Encode(diff.commands());
            retval.tablesThatMustBeEmpty = diff.tablesThatMustBeEmpty();
            retval.reasonsForEmptyTables = diff.reasonsWhyTablesMustBeEmpty();
            retval.requiresSnapshotIsolation = diff.requiresSnapshotIsolation();
            retval.worksWithElastic = diff.worksWithElastic();
        }
        catch (Exception e) {
            String msg = "Unexpected error in adhoc or catalog update: " + e.getClass() + ", " +
                e.getMessage();
            compilerLog.warn(msg, e);
            retval.encodedDiffCommands = null;
            retval.errorMsg = msg;
        }

        return retval;
    }

    /**
     * Append the supplied adhoc DDL to the current catalog's DDL and recompile the
     * jarfile
     */
    private byte[] addDDLToCatalog(byte[] oldCatalogBytes, String[] adhocDDLStmts)
    throws IOException
    {
        VoltCompilerReader ddlReader = null;
        try {
            InMemoryJarfile jarfile = CatalogUtil.loadInMemoryJarFile(oldCatalogBytes);
            // Yoink the current cluster catalog's canonical DDL and append the supplied
            // adhoc DDL to it.
            String oldDDL = new String(jarfile.get(VoltCompiler.AUTOGEN_DDL_FILE_NAME),
                    Constants.UTF8ENCODING);
            compilerLog.debug("OLD DDL: " + oldDDL);
            StringBuilder sb = new StringBuilder();
            sb.append(oldDDL);
            sb.append("\n");
            for (String stmt : adhocDDLStmts) {
                sb.append(stmt);
                sb.append(";\n");
            }
            compilerLog.debug("Adhoc-modified DDL:\n" + sb.toString());
            // Put the new DDL back into the InMemoryJarfile we built because that's
            // the artifact the compiler is expecting to work on.  This allows us to preserve any
            // stored procedure classes and the class loader that came with the catalog
            // before we appended to it.
            ddlReader =
                new VoltCompilerStringReader(VoltCompiler.AUTOGEN_DDL_FILE_NAME, sb.toString());
            ddlReader.putInJar(jarfile, VoltCompiler.AUTOGEN_DDL_FILE_NAME);
            VoltCompiler compiler = new VoltCompiler();
            compiler.compileInMemoryJarfile(jarfile);
            return jarfile.getFullJarBytes();
        }
        finally {
            if (ddlReader != null) {
                try {
                    ddlReader.close();
                }
                catch (IOException ioe) {}
            }
        }
    }

    private byte[] modifyCatalogClasses(byte[] oldCatalogBytes, String deletePatterns,
            byte[] newClassBytes) throws IOException
    {
        // Create a new InMemoryJarfile based on the original catalog bytes,
        // modify it in place based on the @UpdateClasses inputs, and then
        // recompile it if necessary
        InMemoryJarfile jarfile = CatalogUtil.loadInMemoryJarFile(oldCatalogBytes);
        boolean deletedClasses = false;
        if (deletePatterns != null) {
            String[] patterns = deletePatterns.split(",");
            ClassMatcher matcher = new ClassMatcher();
            // Need to concatenate all the classnames together for ClassMatcher
            String currentClasses = "";
            for (String classname : jarfile.getLoader().getClassNames()) {
                currentClasses = currentClasses.concat(classname + "\n");
            }
            matcher.m_classList = currentClasses;
            for (String pattern : patterns) {
                ClassNameMatchStatus status = matcher.addPattern(pattern.trim());
                if (status == ClassNameMatchStatus.MATCH_FOUND) {
                    deletedClasses = true;
                }
            }
            for (String classname : matcher.getMatchedClassList()) {
                jarfile.removeClassFromJar(classname);
            }
        }
        boolean foundClasses = false;
        if (newClassBytes != null) {
            InMemoryJarfile newJarfile = new InMemoryJarfile(newClassBytes);
            for (Entry<String, byte[]> e : newJarfile.entrySet()) {
                String filename = e.getKey();
                if (!filename.endsWith(".class")) {
                    continue;
                }
                foundClasses = true;
                jarfile.put(e.getKey(), e.getValue());
            }
        }
        if (deletedClasses || foundClasses) {
            VoltCompiler compiler = new VoltCompiler();
            compiler.compileInMemoryJarfile(jarfile);
        }
        return jarfile.getFullJarBytes();
    }
}
TOP

Related Classes of org.voltdb.compiler.AsyncCompilerAgentHelper

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.