hu.bme.mit.trainbenchmark.generator.sql.SQLGenerator.java Source code

Java tutorial

Introduction

Here is the source code for hu.bme.mit.trainbenchmark.generator.sql.SQLGenerator.java

Source

/*******************************************************************************
 * Copyright (c) 2010-2015, Benedek Izso, Gabor Szarnyas, Istvan Rath and Daniel Varro
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 *
 * Contributors:
 *   Benedek Izso - initial API and implementation
 *   Gabor Szarnyas - initial API and implementation
 *******************************************************************************/

package hu.bme.mit.trainbenchmark.generator.sql;

import static hu.bme.mit.trainbenchmark.constants.ModelConstants.ID;
import static hu.bme.mit.trainbenchmark.constants.ModelConstants.SENSOR_EDGE;
import static hu.bme.mit.trainbenchmark.constants.ModelConstants.TRACKELEMENT;
import static hu.bme.mit.trainbenchmark.constants.ModelConstants.ancestors;
import static hu.bme.mit.trainbenchmark.sql.constants.SQLConstants.USER;
import hu.bme.mit.trainbenchmark.generator.Generator;
import hu.bme.mit.trainbenchmark.generator.sql.config.SQLGeneratorConfig;
import hu.bme.mit.trainbenchmark.sql.process.MySQLProcess;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.apache.commons.cli.ParseException;
import org.apache.commons.io.FileUtils;

import com.google.common.collect.ImmutableMap;

public class SQLGenerator extends Generator {

    protected SQLGeneratorConfig sqlGeneratorConfig;
    protected String sqlRawPath;
    protected String sqlDumpPath;
    protected String sqlPostgreDumpPath;

    public SQLGenerator(final String[] args) throws ParseException {
        super();
        generatorConfig = sqlGeneratorConfig = new SQLGeneratorConfig(args);
    }

    protected static final Map<String, String> EDGE_TABLE = ImmutableMap.of(SENSOR_EDGE, TRACKELEMENT);

    @Override
    protected String syntax() {
        return "SQL";
    }

    protected BufferedWriter writer;

    public void write(final String s) throws IOException {
        writer.write(s + "\n");
    }

    @Override
    public void initModel() throws IOException {
        // header file (DDL operations)
        final String headerFileDirectory = generatorConfig.getWorkspacePath()
                + "/hu.bme.mit.trainbenchmark.sql/src/main/resources/metamodel/";
        final String headerFilePath = headerFileDirectory
                + (sqlGeneratorConfig.isMemSQL() ? "railway-header-memsql.sql" : "railway-header.sql");
        final File headerFile = new File(headerFilePath);

        // destination file
        sqlRawPath = generatorConfig.getModelPathNameWithoutExtension() + "-raw.sql";
        sqlDumpPath = generatorConfig.getModelPathNameWithoutExtension() + ".sql";
        sqlPostgreDumpPath = generatorConfig.getModelPathNameWithoutExtension() + "-posgres.sql";
        final File sqlRawFile = new File(sqlRawPath);

        // this overwrites the destination file if it exists
        FileUtils.copyFile(headerFile, sqlRawFile);

        writer = new BufferedWriter(new FileWriter(sqlRawFile, true));
    }

    @Override
    protected void persistModel() throws IOException, InterruptedException {
        final String footerFilePath = generatorConfig.getWorkspacePath()
                + "/hu.bme.mit.trainbenchmark.sql/src/main/resources/metamodel/railway-footer.sql";
        final File footerFile = new File(footerFilePath);

        final List<String> lines = FileUtils.readLines(footerFile);
        for (final String line : lines) {
            write(line);
        }

        writer.close();

        compact();
    }

    public void compact() throws IOException, InterruptedException {
        MySQLProcess.stopSQLProcess();
        MySQLProcess.startSQLProcess();

        final Runtime rt = Runtime.getRuntime();
        final String[] commandLoad = { "/bin/bash", "-c", "mysql -u " + USER + " < " + sqlRawPath };
        final Process processLoad = rt.exec(commandLoad);
        processLoad.waitFor();

        final String[] commandDump = { "/bin/bash", "-c",
                "mysqldump -u " + USER + " --databases trainbenchmark --skip-dump-date > " + sqlDumpPath };
        final Process processDump = rt.exec(commandDump);
        processDump.waitFor();
    }

    @Override
    protected Object createVertex(final int id, final String type, final Map<String, Object> attributes,
            final Map<String, Object> outgoingEdges, final Map<String, Object> incomingEdges) throws IOException {
        final StringBuilder columns = new StringBuilder();
        final StringBuilder values = new StringBuilder();

        columns.append("\"" + ID + "\"");
        values.append(id);

        structuralFeaturesToSQL(attributes, columns, values);
        structuralFeaturesToSQL(outgoingEdges, columns, values);
        structuralFeaturesToSQL(incomingEdges, columns, values);

        if (ancestors.containsKey(type)) {
            final String ancestorType = ancestors.get(type);
            write(String.format("INSERT INTO \"%s\" (%s) VALUES (%s);", ancestorType, ID, id));
            write(String.format("INSERT INTO \"%s\" (%s) VALUES (%s);", type, columns.toString(),
                    values.toString()));
        } else {
            final String insertQuery = String.format("INSERT INTO \"%s\" (%s) VALUES (%s);", type,
                    columns.toString(), values.toString());
            write(insertQuery.toString());
        }

        return id;
    }

    @Override
    protected void createEdge(final String label, final Object from, final Object to) throws IOException {
        if (from == null || to == null) {
            return;
        }

        String insertQuery;
        if (SENSOR_EDGE.equals(label)) {
            insertQuery = String.format("UPDATE \"%s\" SET \"%s\" = %s WHERE \"%s\" = %s;", TRACKELEMENT,
                    SENSOR_EDGE, to, ID, from);
        } else {
            insertQuery = String.format("INSERT INTO \"%s\" VALUES (%s, %s);", label, from, to);
        }
        write(insertQuery);
    }

    @Override
    protected void setAttribute(final String type, final Object node, final String key, final Object value)
            throws IOException {
        final String stringValue = valueToString(value);
        final String updateQuery = String.format("UPDATE \"%s\" SET \"%s\" = %s WHERE \"%s\" = %s;", type, key,
                stringValue, ID, node);
        write(updateQuery);
    }

    protected void structuralFeaturesToSQL(final Map<String, Object> attributes, final StringBuilder columns,
            final StringBuilder values) {
        for (final Entry<String, Object> entry : attributes.entrySet()) {
            final String key = entry.getKey();
            final Object value = entry.getValue();

            columns.append(", \"" + key + "\"");
            values.append(", ");

            final String stringValue = (value == null ? "NULL" : valueToString(value));
            values.append(stringValue);
        }
    }

    private String valueToString(final Object value) {
        String stringValue;
        if (value instanceof String) {
            // escape string
            stringValue = "\"" + value + "\"";
        } else if (value instanceof Enum) {
            // change enum to ordinal value
            final Enum<?> enumeration = (Enum<?>) value;
            stringValue = Integer.toString(enumeration.ordinal());
        } else {
            stringValue = value.toString();
        }
        return stringValue;
    }

}