Package org.openstreetmap.osmosis.apidb.v0_6

Source Code of org.openstreetmap.osmosis.apidb.v0_6.ApidbWriter

// This software is released into the Public Domain.  See copying.txt for details.
package org.openstreetmap.osmosis.apidb.v0_6;

import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.sql.Timestamp;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;

import org.openstreetmap.osmosis.apidb.common.DatabaseContext;
import org.openstreetmap.osmosis.apidb.v0_6.impl.ChangesetManager;
import org.openstreetmap.osmosis.apidb.v0_6.impl.MemberTypeRenderer;
import org.openstreetmap.osmosis.apidb.v0_6.impl.SchemaVersionValidator;
import org.openstreetmap.osmosis.apidb.v0_6.impl.UserManager;
import org.openstreetmap.osmosis.core.OsmosisRuntimeException;
import org.openstreetmap.osmosis.core.container.v0_6.BoundContainer;
import org.openstreetmap.osmosis.core.container.v0_6.EntityContainer;
import org.openstreetmap.osmosis.core.container.v0_6.EntityProcessor;
import org.openstreetmap.osmosis.core.container.v0_6.NodeContainer;
import org.openstreetmap.osmosis.core.container.v0_6.RelationContainer;
import org.openstreetmap.osmosis.core.container.v0_6.WayContainer;
import org.openstreetmap.osmosis.core.database.DatabaseLoginCredentials;
import org.openstreetmap.osmosis.core.database.DatabasePreferences;
import org.openstreetmap.osmosis.core.database.DbFeature;
import org.openstreetmap.osmosis.core.database.DbFeatureHistory;
import org.openstreetmap.osmosis.core.database.DbOrderedFeature;
import org.openstreetmap.osmosis.core.domain.v0_6.Entity;
import org.openstreetmap.osmosis.core.domain.v0_6.Node;
import org.openstreetmap.osmosis.core.domain.v0_6.Relation;
import org.openstreetmap.osmosis.core.domain.v0_6.RelationMember;
import org.openstreetmap.osmosis.core.domain.v0_6.Tag;
import org.openstreetmap.osmosis.core.domain.v0_6.Way;
import org.openstreetmap.osmosis.core.domain.v0_6.WayNode;
import org.openstreetmap.osmosis.core.task.v0_6.Sink;
import org.openstreetmap.osmosis.core.util.FixedPrecisionCoordinateConvertor;
import org.openstreetmap.osmosis.core.util.TileCalculator;


/**
* An OSM data sink for storing all data to a database. This task is intended for writing to an
* empty database.
*
* @author Brett Henderson
*/
public class ApidbWriter implements Sink, EntityProcessor {

    // These SQL strings are the prefix to statements that will be built based
    // on how many rows of data are to be inserted at a time.
  private static final String INSERT_SQL_NODE_COLUMNS =
    "INSERT INTO nodes(node_id, timestamp, version, visible, changeset_id, latitude, longitude, tile)";
  private static final String INSERT_SQL_NODE_PARAMS = "?, ?, ?, ?, ?, ?, ?, ?";
  private static final int INSERT_PRM_COUNT_NODE = 8;

    private static final String INSERT_SQL_NODE_TAG_COLUMNS = "INSERT INTO node_tags (node_id, k, v, version)";
    private static final String INSERT_SQL_NODE_TAG_PARAMS = "?, ?, ?, ?";
  private static final int INSERT_PRM_COUNT_NODE_TAG = 4;

    private static final String INSERT_SQL_WAY_COLUMNS =
      "INSERT INTO ways (way_id, timestamp, version, visible, changeset_id)";
    private static final String INSERT_SQL_WAY_PARAMS = "?, ?, ?, ?, ?";
  private static final int INSERT_PRM_COUNT_WAY = 5;

    private static final String INSERT_SQL_WAY_TAG_COLUMNS = "INSERT INTO way_tags (way_id, k, v, version)";
    private static final String INSERT_SQL_WAY_TAG_PARAMS = "?, ?, ?, ?";
  private static final int INSERT_PRM_COUNT_WAY_TAG = 4;

    private static final String INSERT_SQL_WAY_NODE_COLUMNS =
      "INSERT INTO way_nodes (way_id, node_id, sequence_id, version)";
    private static final String INSERT_SQL_WAY_NODE_PARAMS = "?, ?, ?, ?";
  private static final int INSERT_PRM_COUNT_WAY_NODE = 4;

    private static final String INSERT_SQL_RELATION_COLUMNS =
      "INSERT INTO relations (relation_id, timestamp, version, visible, changeset_id)";
    private static final String INSERT_SQL_RELATION_PARAMS =
      "?, ?, ?, ?, ?";
  private static final int INSERT_PRM_COUNT_RELATION = 5;

    private static final String INSERT_SQL_RELATION_TAG_COLUMNS =
        "INSERT INTO relation_tags (relation_id, k, v, version)";
    private static final String INSERT_SQL_RELATION_TAG_PARAMS = "?, ?, ?, ?";
  private static final int INSERT_PRM_COUNT_RELATION_TAG = 4;

    private static final String INSERT_SQL_RELATION_MEMBER_COLUMNS =
      "INSERT INTO relation_members (relation_id, member_type, member_id, sequence_id, member_role, version)";
    private static final String INSERT_SQL_RELATION_MEMBER_PARAMS_MYSQL =
      "?, ?, ?, ?, ?, ?";
    private static final String INSERT_SQL_RELATION_MEMBER_PARAMS_PGSQL =
      "?, ?::nwr_enum, ?, ?, ?, ?";
  private static final int INSERT_PRM_COUNT_RELATION_MEMBER = 6;

    // These tables will have indexes disabled during loading data.
    private static final List<String> DISABLE_KEY_TABLES = Arrays.asList(new String[] {"nodes",
            "node_tags", "ways", "way_tags",
            "way_nodes", "relations",
            "relation_tags", "relation_members"});

    // These SQL statements will be invoked after loading history tables to
    // populate the current tables.
    private static final int LOAD_CURRENT_NODE_ROW_COUNT = 1000000;

    private static final int LOAD_CURRENT_WAY_ROW_COUNT = 100000;

    private static final int LOAD_CURRENT_RELATION_ROW_COUNT = 100000;

    private static final String LOAD_CURRENT_NODES =
      "INSERT INTO current_nodes SELECT node_id, latitude, longitude, changeset_id, visible, timestamp, tile, version"
            + " FROM nodes WHERE node_id >= ? AND node_id < ?";

    private static final String LOAD_CURRENT_NODE_TAGS =
      "INSERT INTO current_node_tags SELECT node_id, k, v FROM node_tags WHERE node_id >= ? AND node_id < ?";

    private static final String LOAD_CURRENT_WAYS =
      "INSERT INTO current_ways SELECT way_id, changeset_id, timestamp, visible, version FROM ways"
            + " WHERE way_id >= ? AND way_id < ?";

    private static final String LOAD_CURRENT_WAY_TAGS =
      "INSERT INTO current_way_tags SELECT way_id, k, v FROM way_tags"
            + " WHERE way_id >= ? AND way_id < ?";

    private static final String LOAD_CURRENT_WAY_NODES =
      "INSERT INTO current_way_nodes SELECT way_id, node_id, sequence_id FROM way_nodes"
            + " WHERE way_id >= ? AND way_id < ?";

    private static final String LOAD_CURRENT_RELATIONS =
      "INSERT INTO current_relations SELECT relation_id, changeset_id, timestamp, visible, version"
            + " FROM relations WHERE relation_id >= ? AND relation_id < ?";

    private static final String LOAD_CURRENT_RELATION_TAGS =
      "INSERT INTO current_relation_tags SELECT relation_id, k, v FROM relation_tags"
            + " WHERE relation_id >= ? AND relation_id < ?";

    private static final String LOAD_CURRENT_RELATION_MEMBERS =
      "INSERT INTO current_relation_members (relation_id, member_id, member_role, member_type, sequence_id)"
        + " SELECT relation_id, member_id, member_role, member_type, sequence_id"
            + " FROM relation_members WHERE relation_id >= ? AND relation_id < ?";

    // These tables will be locked for exclusive access while loading data.
  private static final List<String> LOCK_TABLES = Arrays.asList(new String[] {"nodes", "node_tags", "ways",
      "way_tags", "way_nodes", "relations", "relation_tags", "relation_members", "current_nodes",
      "current_node_tags", "current_ways", "current_way_tags", "current_way_nodes", "current_relations",
      "current_relation_tags", "current_relation_members", "users", "changesets", "changeset_tags" });

    // These constants define how many rows of each data type will be inserted
    // with single insert statements.
    private static final int INSERT_BULK_ROW_COUNT_NODE = 100;
    private static final int INSERT_BULK_ROW_COUNT_NODE_TAG = 100;
    private static final int INSERT_BULK_ROW_COUNT_WAY = 100;
    private static final int INSERT_BULK_ROW_COUNT_WAY_TAG = 100;
    private static final int INSERT_BULK_ROW_COUNT_WAY_NODE = 100;
    private static final int INSERT_BULK_ROW_COUNT_RELATION = 100;
    private static final int INSERT_BULK_ROW_COUNT_RELATION_TAG = 100;
    private static final int INSERT_BULK_ROW_COUNT_RELATION_MEMBER = 100;

    /**
   * Builds a multi-row SQL insert statement.
   *
   * @param columnSql
   *            The basic query without value bind variables.
   * @param parametersSql
   *            The SQL parameters portion of the query.
   * @param rowCount
   *            The number of rows to insert in a single query.
   * @return The generated SQL statement.
   */
    private static String buildSqlInsertStatement(String columnSql, String parametersSql, int rowCount) {
        StringBuilder buffer;

        buffer = new StringBuilder();

        buffer.append(columnSql).append(" VALUES ");

        for (int i = 0; i < rowCount; i++) {
            if (i > 0) {
                buffer.append(", ");
            }
           
            buffer.append("(");
            buffer.append(parametersSql);
            buffer.append(")");
        }

        return buffer.toString();
    }

   
    private String insertSqlSingleNode;
    private String insertSqlBulkNode;
    private String insertSqlSingleNodeTag;
    private String insertSqlBulkNodeTag;
    private String insertSqlSingleWay;
    private String insertSqlBulkWay;
    private String insertSqlSingleWayTag;
    private String insertSqlBulkWayTag;
    private String insertSqlSingleWayNode;
    private String insertSqlBulkWayNode;
    private String insertSqlSingleRelation;
    private String insertSqlBulkRelation;
    private String insertSqlSingleRelationTag;
    private String insertSqlBulkRelationTag;
    private String insertSqlSingleRelationMember;
    private String insertSqlBulkRelationMember;
    private final DatabaseContext dbCtx;
    private final UserManager userManager;
    private final ChangesetManager changesetManager;
    private final SchemaVersionValidator schemaVersionValidator;
    private final boolean lockTables;
    private final boolean populateCurrentTables;
    private final List<Node> nodeBuffer;
    private final List<DbFeatureHistory<DbFeature<Tag>>> nodeTagBuffer;
    private final List<Way> wayBuffer;
    private final List<DbFeatureHistory<DbFeature<Tag>>> wayTagBuffer;
    private final List<DbFeatureHistory<DbOrderedFeature<WayNode>>> wayNodeBuffer;
    private final List<Relation> relationBuffer;
    private final List<DbFeatureHistory<DbFeature<Tag>>> relationTagBuffer;
    private final List<DbFeatureHistory<DbOrderedFeature<RelationMember>>> relationMemberBuffer;
    private long maxNodeId;
    private long minNodeId;
    private long maxWayId;
    private long minWayId;
    private long maxRelationId;
    private long minRelationId;
    private final TileCalculator tileCalculator;
    private final MemberTypeRenderer memberTypeRenderer;
    private boolean initialized;
    private PreparedStatement singleNodeStatement;
    private PreparedStatement bulkNodeStatement;
    private PreparedStatement singleNodeTagStatement;
    private PreparedStatement bulkNodeTagStatement;
    private PreparedStatement singleWayStatement;
    private PreparedStatement bulkWayStatement;
    private PreparedStatement singleWayTagStatement;
    private PreparedStatement bulkWayTagStatement;
    private PreparedStatement singleWayNodeStatement;
    private PreparedStatement bulkWayNodeStatement;
    private PreparedStatement singleRelationStatement;
    private PreparedStatement bulkRelationStatement;
    private PreparedStatement singleRelationTagStatement;
    private PreparedStatement bulkRelationTagStatement;
    private PreparedStatement singleRelationMemberStatement;
    private PreparedStatement bulkRelationMemberStatement;
    private PreparedStatement loadCurrentNodesStatement;
    private PreparedStatement loadCurrentNodeTagsStatement;
    private PreparedStatement loadCurrentWaysStatement;
    private PreparedStatement loadCurrentWayTagsStatement;
  private PreparedStatement loadCurrentWayNodesStatement;
  private PreparedStatement loadCurrentRelationsStatement;
  private PreparedStatement loadCurrentRelationTagsStatement;
  private PreparedStatement loadCurrentRelationMembersStatement;

    /**
     * Creates a new instance.
     *
     * @param loginCredentials Contains all information required to connect to the database.
     * @param preferences Contains preferences configuring database behaviour.
     * @param lockTables If true, all tables will be locked during loading.
     * @param populateCurrentTables If true, the current tables will be populated as well as history
     *        tables.
     */
    public ApidbWriter(DatabaseLoginCredentials loginCredentials, DatabasePreferences preferences, boolean lockTables,
            boolean populateCurrentTables) {
        dbCtx = new DatabaseContext(loginCredentials);
       
        userManager = new UserManager(dbCtx);
        changesetManager = new ChangesetManager(dbCtx);

        schemaVersionValidator = new SchemaVersionValidator(loginCredentials, preferences);

        this.lockTables = lockTables;
        this.populateCurrentTables = populateCurrentTables;

        nodeBuffer = new ArrayList<Node>();
        nodeTagBuffer = new ArrayList<DbFeatureHistory<DbFeature<Tag>>>();
        wayBuffer = new ArrayList<Way>();
        wayTagBuffer = new ArrayList<DbFeatureHistory<DbFeature<Tag>>>();
        wayNodeBuffer = new ArrayList<DbFeatureHistory<DbOrderedFeature<WayNode>>>();
        relationBuffer = new ArrayList<Relation>();
        relationTagBuffer = new ArrayList<DbFeatureHistory<DbFeature<Tag>>>();
        relationMemberBuffer = new ArrayList<DbFeatureHistory<DbOrderedFeature<RelationMember>>>();

        maxNodeId = Long.MIN_VALUE;
        minNodeId = Long.MAX_VALUE;
        maxWayId = Long.MIN_VALUE;
        minWayId = Long.MAX_VALUE;
        maxRelationId = Long.MIN_VALUE;
        minRelationId = Long.MAX_VALUE;

        tileCalculator = new TileCalculator();
        memberTypeRenderer = new MemberTypeRenderer();

        initialized = false;
    }
   
   
    private void buildSqlStatements() {
      insertSqlSingleNode = buildSqlInsertStatement(INSERT_SQL_NODE_COLUMNS, INSERT_SQL_NODE_PARAMS, 1);
    insertSqlBulkNode = buildSqlInsertStatement(INSERT_SQL_NODE_COLUMNS, INSERT_SQL_NODE_PARAMS,
        INSERT_BULK_ROW_COUNT_NODE);
    insertSqlSingleNodeTag = buildSqlInsertStatement(
        INSERT_SQL_NODE_TAG_COLUMNS, INSERT_SQL_NODE_TAG_PARAMS, 1);
    insertSqlBulkNodeTag = buildSqlInsertStatement(INSERT_SQL_NODE_TAG_COLUMNS, INSERT_SQL_NODE_TAG_PARAMS,
        INSERT_BULK_ROW_COUNT_NODE_TAG);
    insertSqlSingleWay = buildSqlInsertStatement(INSERT_SQL_WAY_COLUMNS, INSERT_SQL_WAY_PARAMS, 1);
    insertSqlBulkWay = buildSqlInsertStatement(INSERT_SQL_WAY_COLUMNS, INSERT_SQL_WAY_PARAMS,
        INSERT_BULK_ROW_COUNT_WAY);
    insertSqlSingleWayTag = buildSqlInsertStatement(INSERT_SQL_WAY_TAG_COLUMNS, INSERT_SQL_WAY_TAG_PARAMS, 1);
    insertSqlBulkWayTag = buildSqlInsertStatement(INSERT_SQL_WAY_TAG_COLUMNS, INSERT_SQL_WAY_TAG_PARAMS,
        INSERT_BULK_ROW_COUNT_WAY_TAG);
    insertSqlSingleWayNode = buildSqlInsertStatement(
        INSERT_SQL_WAY_NODE_COLUMNS, INSERT_SQL_WAY_NODE_PARAMS, 1);
    insertSqlBulkWayNode = buildSqlInsertStatement(INSERT_SQL_WAY_NODE_COLUMNS, INSERT_SQL_WAY_NODE_PARAMS,
        INSERT_BULK_ROW_COUNT_WAY_NODE);
    insertSqlSingleRelation = buildSqlInsertStatement(INSERT_SQL_RELATION_COLUMNS, INSERT_SQL_RELATION_PARAMS,
        1);
    insertSqlBulkRelation = buildSqlInsertStatement(INSERT_SQL_RELATION_COLUMNS, INSERT_SQL_RELATION_PARAMS,
        INSERT_BULK_ROW_COUNT_RELATION);
      insertSqlSingleRelationTag = buildSqlInsertStatement(INSERT_SQL_RELATION_TAG_COLUMNS,
        INSERT_SQL_RELATION_TAG_PARAMS, 1);
    insertSqlBulkRelationTag = buildSqlInsertStatement(INSERT_SQL_RELATION_TAG_COLUMNS,
        INSERT_SQL_RELATION_TAG_PARAMS, INSERT_BULK_ROW_COUNT_RELATION_TAG);
    }
   

    /**
     * Initialises prepared statements and obtains database locks. Can be called multiple times.
     */
    private void initialize() {
        if (!initialized) {
            schemaVersionValidator.validateVersion(ApidbVersionConstants.SCHEMA_MIGRATIONS);
           
            buildSqlStatements();
           
            switch (dbCtx.getDatabaseType()) {
            case POSTGRESQL:
          insertSqlSingleRelationMember = buildSqlInsertStatement(INSERT_SQL_RELATION_MEMBER_COLUMNS,
              INSERT_SQL_RELATION_MEMBER_PARAMS_PGSQL, 1);
          insertSqlBulkRelationMember = buildSqlInsertStatement(INSERT_SQL_RELATION_MEMBER_COLUMNS,
              INSERT_SQL_RELATION_MEMBER_PARAMS_PGSQL, INSERT_BULK_ROW_COUNT_RELATION_MEMBER);
                break;
            case MYSQL:
          insertSqlSingleRelationMember = buildSqlInsertStatement(INSERT_SQL_RELATION_MEMBER_COLUMNS,
              INSERT_SQL_RELATION_MEMBER_PARAMS_MYSQL, 1);
          insertSqlBulkRelationMember = buildSqlInsertStatement(INSERT_SQL_RELATION_MEMBER_COLUMNS,
              INSERT_SQL_RELATION_MEMBER_PARAMS_MYSQL, INSERT_BULK_ROW_COUNT_RELATION_MEMBER);
                break;
            default:
                throw new OsmosisRuntimeException("Unknown database type " + dbCtx.getDatabaseType() + ".");
            }

            bulkNodeStatement = dbCtx.prepareStatement(insertSqlBulkNode);
            singleNodeStatement = dbCtx.prepareStatement(insertSqlSingleNode);
            bulkNodeTagStatement = dbCtx.prepareStatement(insertSqlBulkNodeTag);
            singleNodeTagStatement = dbCtx.prepareStatement(insertSqlSingleNodeTag);
            bulkWayStatement = dbCtx.prepareStatement(insertSqlBulkWay);
            singleWayStatement = dbCtx.prepareStatement(insertSqlSingleWay);
            bulkWayTagStatement = dbCtx.prepareStatement(insertSqlBulkWayTag);
            singleWayTagStatement = dbCtx.prepareStatement(insertSqlSingleWayTag);
            bulkWayNodeStatement = dbCtx.prepareStatement(insertSqlBulkWayNode);
            singleWayNodeStatement = dbCtx.prepareStatement(insertSqlSingleWayNode);
            bulkRelationStatement = dbCtx.prepareStatement(insertSqlBulkRelation);
            singleRelationStatement = dbCtx.prepareStatement(insertSqlSingleRelation);
            bulkRelationTagStatement = dbCtx.prepareStatement(insertSqlBulkRelationTag);
            singleRelationTagStatement = dbCtx.prepareStatement(insertSqlSingleRelationTag);
            bulkRelationMemberStatement = dbCtx.prepareStatement(insertSqlBulkRelationMember);
            singleRelationMemberStatement = dbCtx.prepareStatement(insertSqlSingleRelationMember);

            loadCurrentNodesStatement = dbCtx.prepareStatement(LOAD_CURRENT_NODES);
            loadCurrentNodeTagsStatement = dbCtx.prepareStatement(LOAD_CURRENT_NODE_TAGS);
            loadCurrentWaysStatement = dbCtx.prepareStatement(LOAD_CURRENT_WAYS);
            loadCurrentWayTagsStatement = dbCtx.prepareStatement(LOAD_CURRENT_WAY_TAGS);
            loadCurrentWayNodesStatement = dbCtx.prepareStatement(LOAD_CURRENT_WAY_NODES);
            loadCurrentRelationsStatement = dbCtx.prepareStatement(LOAD_CURRENT_RELATIONS);
            loadCurrentRelationTagsStatement = dbCtx.prepareStatement(LOAD_CURRENT_RELATION_TAGS);
            loadCurrentRelationMembersStatement = dbCtx.prepareStatement(LOAD_CURRENT_RELATION_MEMBERS);

            // Disable indexes to improve load performance.
            dbCtx.disableIndexes(DISABLE_KEY_TABLES);

            // Lock tables if required to improve load performance.
            if (lockTables) {
              dbCtx.lockTables(LOCK_TABLES);
            }

            initialized = true;
        }
    }

    /**
     * Sets node values as bind variable parameters to a node insert query.
     *
     * @param statement The prepared statement to add the values to.
     * @param initialIndex The offset index of the first variable to set.
     * @param node The node containing the data to be inserted.
     */
    private void populateNodeParameters(PreparedStatement statement, int initialIndex, Node node) {
        int prmIndex;

        prmIndex = initialIndex;

        // We can't write an entity with a null timestamp.
        if (node.getTimestamp() == null) {
            throw new OsmosisRuntimeException("Node " + node.getId() + " does not have a timestamp set.");
        }

        try {
            statement.setLong(prmIndex++, node.getId());
            statement.setTimestamp(prmIndex++, new Timestamp(node.getTimestamp().getTime()));
            statement.setInt(prmIndex++, node.getVersion());
            statement.setBoolean(prmIndex++, true);
            statement.setLong(prmIndex++, node.getChangesetId());
            statement.setInt(prmIndex++, FixedPrecisionCoordinateConvertor.convertToFixed(node.getLatitude()));
            statement.setInt(prmIndex++, FixedPrecisionCoordinateConvertor.convertToFixed(node.getLongitude()));
            statement.setLong(prmIndex++, tileCalculator.calculateTile(node.getLatitude(), node.getLongitude()));

        } catch (SQLException e) {
            throw new OsmosisRuntimeException("Unable to set a prepared statement parameter for a node.", e);
        }
    }

    /**
     * Sets way values as bind variable parameters to a way insert query.
     *
     * @param statement The prepared statement to add the values to.
     * @param initialIndex The offset index of the first variable to set.
     * @param way The way containing the data to be inserted.
     */
    private void populateWayParameters(PreparedStatement statement, int initialIndex, Way way) {
        int prmIndex;

        prmIndex = initialIndex;

        // We can't write an entity with a null timestamp.
        if (way.getTimestamp() == null) {
            throw new OsmosisRuntimeException("Way " + way.getId() + " does not have a timestamp set.");
        }

        try {
            statement.setLong(prmIndex++, way.getId());
            statement.setTimestamp(prmIndex++, new Timestamp(way.getTimestamp().getTime()));
            statement.setInt(prmIndex++, way.getVersion());
            statement.setBoolean(prmIndex++, true);
            statement.setLong(prmIndex++, way.getChangesetId());

        } catch (SQLException e) {
            throw new OsmosisRuntimeException("Unable to set a prepared statement parameter for a way.", e);
        }
    }

    /**
     * Sets tag values as bind variable parameters to a tag insert query.
     *
     * @param statement The prepared statement to add the values to.
     * @param initialIndex The offset index of the first variable to set.
     * @param dbEntityTag The entity tag containing the data to be inserted.
     */
    private void populateEntityTagParameters(PreparedStatement statement, int initialIndex,
        DbFeatureHistory<DbFeature<Tag>> dbEntityTag) {
        int prmIndex;
        Tag tag;

        prmIndex = initialIndex;

        tag = dbEntityTag.getFeature().getFeature();

        try {
            statement.setLong(prmIndex++, dbEntityTag.getFeature().getEntityId());
            statement.setString(prmIndex++, tag.getKey());
            statement.setString(prmIndex++, tag.getValue());
            statement.setInt(prmIndex++, dbEntityTag.getVersion());

        } catch (SQLException e) {
            throw new OsmosisRuntimeException("Unable to set a prepared statement parameter for an entity tag.", e);
        }
    }

    /**
     * Sets node reference values as bind variable parameters to a way node insert query.
     *
     * @param statement The prepared statement to add the values to.
     * @param initialIndex The offset index of the first variable to set.
     * @param dbWayNode The way node containing the data to be inserted.
     */
    private void populateWayNodeParameters(PreparedStatement statement, int initialIndex,
        DbFeatureHistory<DbOrderedFeature<WayNode>> dbWayNode) {
        int prmIndex;

        prmIndex = initialIndex;

        try {
            statement.setLong(prmIndex++, dbWayNode.getFeature().getEntityId());
            statement.setLong(prmIndex++, dbWayNode.getFeature().getFeature().getNodeId());
            statement.setInt(prmIndex++, dbWayNode.getFeature().getSequenceId());
            statement.setInt(prmIndex++, dbWayNode.getVersion());

        } catch (SQLException e) {
            throw new OsmosisRuntimeException("Unable to set a prepared statement parameter for a way node.", e);
        }
    }

    /**
     * Sets relation values as bind variable parameters to a relation insert query.
     *
     * @param statement The prepared statement to add the values to.
     * @param initialIndex The offset index of the first variable to set.
     * @param relation The way containing the data to be inserted.
     */
    private void populateRelationParameters(PreparedStatement statement, int initialIndex, Relation relation) {
        int prmIndex;

        prmIndex = initialIndex;

        // We can't write an entity with a null timestamp.
        if (relation.getTimestamp() == null) {
            throw new OsmosisRuntimeException("Relation " + relation.getId() + " does not have a timestamp set.");
        }

        try {
            statement.setLong(prmIndex++, relation.getId());
            statement.setTimestamp(prmIndex++, new Timestamp(relation.getTimestamp().getTime()));
            statement.setInt(prmIndex++, relation.getVersion());
            statement.setBoolean(prmIndex++, true);
            statement.setLong(prmIndex++, relation.getChangesetId());

        } catch (SQLException e) {
            throw new OsmosisRuntimeException("Unable to set a prepared statement parameter for a relation.", e);
        }
    }

    /**
     * Sets member reference values as bind variable parameters to a relation member insert query.
     *
     * @param statement The prepared statement to add the values to.
     * @param initialIndex The offset index of the first variable to set.
     * @param dbRelationMember The relation member containing the data to be inserted.
     */
    private void populateRelationMemberParameters(PreparedStatement statement, int initialIndex,
        DbFeatureHistory<DbOrderedFeature<RelationMember>> dbRelationMember) {
        int prmIndex;
        RelationMember relationMember;

        prmIndex = initialIndex;

        relationMember = dbRelationMember.getFeature().getFeature();

        try {
            statement.setLong(prmIndex++, dbRelationMember.getFeature().getEntityId());
            statement.setString(prmIndex++, memberTypeRenderer.render(relationMember.getMemberType()));
            statement.setLong(prmIndex++, relationMember.getMemberId());
            statement.setInt(prmIndex++, dbRelationMember.getFeature().getSequenceId());
            statement.setString(prmIndex++, relationMember.getMemberRole());
            statement.setInt(prmIndex++, dbRelationMember.getVersion());

        } catch (SQLException e) {
            throw new OsmosisRuntimeException("Unable to set a prepared statement parameter for a relation member.", e);
        }
    }

    /**
     * Flushes nodes to the database. If complete is false, this will only write nodes until the
     * remaining node count is less than the multi-row insert statement row count. If complete is
     * true, all remaining rows will be written using single row insert statements.
     *
     * @param complete If true, all data will be written to the database. If false, some data may be
     *        left until more data is available.
     */
    private void flushNodes(boolean complete) {
        while (nodeBuffer.size() >= INSERT_BULK_ROW_COUNT_NODE) {
            int prmIndex;
            List<Node> processedNodes;

            processedNodes = new ArrayList<Node>(INSERT_BULK_ROW_COUNT_NODE);

            prmIndex = 1;
            for (int i = 0; i < INSERT_BULK_ROW_COUNT_NODE; i++) {
                Node node;

                node = nodeBuffer.remove(0);
                processedNodes.add(node);

                populateNodeParameters(bulkNodeStatement, prmIndex, node);
                prmIndex += INSERT_PRM_COUNT_NODE;
            }

            try {
                bulkNodeStatement.executeUpdate();
            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to bulk insert nodes into the database.", e);
            }

            for (Node node : processedNodes) {
                addNodeTags(node);
            }
        }

        if (complete) {
            while (nodeBuffer.size() > 0) {
                Node node;

                node = nodeBuffer.remove(0);

                populateNodeParameters(singleNodeStatement, 1, node);

                try {
                    singleNodeStatement.executeUpdate();
                } catch (SQLException e) {
                    throw new OsmosisRuntimeException("Unable to insert a node into the database.", e);
                }

                addNodeTags(node);
            }
        }
    }

    /**
     * Flushes node tags to the database. If complete is false, this will only write node tags until
     * the remaining node tag count is less than the multi-row insert statement row count. If
     * complete is true, all remaining rows will be written using single row insert statements.
     *
     * @param complete If true, all data will be written to the database. If false, some data may be
     *        left until more data is available.
     */
    private void flushNodeTags(boolean complete) {
        while (nodeTagBuffer.size() >= INSERT_BULK_ROW_COUNT_NODE_TAG) {
            int prmIndex;

            prmIndex = 1;
            for (int i = 0; i < INSERT_BULK_ROW_COUNT_NODE_TAG; i++) {
                populateEntityTagParameters(bulkNodeTagStatement, prmIndex, nodeTagBuffer.remove(0));
                prmIndex += INSERT_PRM_COUNT_NODE_TAG;
            }

            try {
                bulkNodeTagStatement.executeUpdate();
            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to bulk insert node tags into the database.", e);
            }
        }

        if (complete) {
            while (nodeTagBuffer.size() > 0) {
                populateEntityTagParameters(singleNodeTagStatement, 1, nodeTagBuffer.remove(0));

                try {
                    singleNodeTagStatement.executeUpdate();
                } catch (SQLException e) {
                    throw new OsmosisRuntimeException("Unable to insert a node tag into the database.", e);
                }
            }
        }
    }

    /**
     * Flushes ways to the database. If complete is false, this will only write ways until the
     * remaining way count is less than the multi-row insert statement row count. If complete is
     * true, all remaining rows will be written using single row insert statements.
     *
     * @param complete If true, all data will be written to the database. If false, some data may be
     *        left until more data is available.
     */
    private void flushWays(boolean complete) {
        while (wayBuffer.size() >= INSERT_BULK_ROW_COUNT_WAY) {
            List<Way> processedWays;
            int prmIndex;

            processedWays = new ArrayList<Way>(INSERT_BULK_ROW_COUNT_WAY);

            prmIndex = 1;
            for (int i = 0; i < INSERT_BULK_ROW_COUNT_WAY; i++) {
                Way way;

                way = wayBuffer.remove(0);
                processedWays.add(way);

                populateWayParameters(bulkWayStatement, prmIndex, way);
                prmIndex += INSERT_PRM_COUNT_WAY;
            }

            try {
                bulkWayStatement.executeUpdate();
            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to bulk insert ways into the database.", e);
            }

            for (Way way : processedWays) {
                addWayTags(way);
                addWayNodes(way);
            }
        }

        if (complete) {
            while (wayBuffer.size() > 0) {
                Way way;

                way = wayBuffer.remove(0);

                populateWayParameters(singleWayStatement, 1, way);

                try {
                    singleWayStatement.executeUpdate();
                } catch (SQLException e) {
                    throw new OsmosisRuntimeException("Unable to insert a way into the database.", e);
                }

                addWayTags(way);
                addWayNodes(way);
            }
        }
    }

    /**
     * Flushes way tags to the database. If complete is false, this will only write way tags until
     * the remaining way tag count is less than the multi-row insert statement row count. If
     * complete is true, all remaining rows will be written using single row insert statements.
     *
     * @param complete If true, all data will be written to the database. If false, some data may be
     *        left until more data is available.
     */
    private void flushWayTags(boolean complete) {
        while (wayTagBuffer.size() >= INSERT_BULK_ROW_COUNT_WAY_TAG) {
            int prmIndex;

            prmIndex = 1;
            for (int i = 0; i < INSERT_BULK_ROW_COUNT_WAY_TAG; i++) {
                populateEntityTagParameters(bulkWayTagStatement, prmIndex, wayTagBuffer.remove(0));
                prmIndex += INSERT_PRM_COUNT_WAY_TAG;
            }

            try {
                bulkWayTagStatement.executeUpdate();
            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to bulk insert way tags into the database.", e);
            }
        }

        if (complete) {
            while (wayTagBuffer.size() > 0) {
                populateEntityTagParameters(singleWayTagStatement, 1, wayTagBuffer.remove(0));

                try {
                    singleWayTagStatement.executeUpdate();
                } catch (SQLException e) {
                    throw new OsmosisRuntimeException("Unable to insert a way tag into the database.", e);
                }
            }
        }
    }

    /**
     * Flushes way nodes to the database. If complete is false, this will only write way nodes until
     * the remaining way node count is less than the multi-row insert statement row count. If
     * complete is true, all remaining rows will be written using single row insert statements.
     *
     * @param complete If true, all data will be written to the database. If false, some data may be
     *        left until more data is available.
     */
    private void flushWayNodes(boolean complete) {
        while (wayNodeBuffer.size() >= INSERT_BULK_ROW_COUNT_WAY_NODE) {
            int prmIndex;

            prmIndex = 1;
            for (int i = 0; i < INSERT_BULK_ROW_COUNT_WAY_NODE; i++) {
                populateWayNodeParameters(bulkWayNodeStatement, prmIndex, wayNodeBuffer.remove(0));
                prmIndex += INSERT_PRM_COUNT_WAY_NODE;
            }

            try {
                bulkWayNodeStatement.executeUpdate();
            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to bulk insert way nodes into the database.", e);
            }
        }

        if (complete) {
            while (wayNodeBuffer.size() > 0) {
                populateWayNodeParameters(singleWayNodeStatement, 1, wayNodeBuffer.remove(0));

                try {
                    singleWayNodeStatement.executeUpdate();
                } catch (SQLException e) {
                    throw new OsmosisRuntimeException("Unable to insert a way node into the database.", e);
                }
            }
        }
    }

    /**
     * Flushes relations to the database. If complete is false, this will only write relations until
     * the remaining way count is less than the multi-row insert statement row count. If complete is
     * true, all remaining rows will be written using single row insert statements.
     *
     * @param complete If true, all data will be written to the database. If false, some data may be
     *        left until more data is available.
     */
    private void flushRelations(boolean complete) {
        while (relationBuffer.size() >= INSERT_BULK_ROW_COUNT_RELATION) {
            List<Relation> processedRelations;
            int prmIndex;

            processedRelations = new ArrayList<Relation>(INSERT_BULK_ROW_COUNT_RELATION);

            prmIndex = 1;
            for (int i = 0; i < INSERT_BULK_ROW_COUNT_RELATION; i++) {
                Relation relation;

                relation = relationBuffer.remove(0);
                processedRelations.add(relation);

                populateRelationParameters(bulkRelationStatement, prmIndex, relation);
                prmIndex += INSERT_PRM_COUNT_RELATION;
            }

            try {
                bulkRelationStatement.executeUpdate();
            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to bulk insert relations into the database.", e);
            }

            for (Relation relation : processedRelations) {
                addRelationTags(relation);
                addRelationMembers(relation);
            }
        }

        if (complete) {
            while (relationBuffer.size() > 0) {
                Relation relation;

                relation = relationBuffer.remove(0);

                populateRelationParameters(singleRelationStatement, 1, relation);

                try {
                    singleRelationStatement.executeUpdate();
                } catch (SQLException e) {
                    throw new OsmosisRuntimeException("Unable to insert a relation into the database.", e);
                }

                addRelationTags(relation);
                addRelationMembers(relation);
            }
        }
    }

    /**
     * Flushes relation tags to the database. If complete is false, this will only write relation
     * tags until the remaining relation tag count is less than the multi-row insert statement row
     * count. If complete is true, all remaining rows will be written using single row insert
     * statements.
     *
     * @param complete If true, all data will be written to the database. If false, some data may be
     *        left until more data is available.
     */
    private void flushRelationTags(boolean complete) {
        while (relationTagBuffer.size() >= INSERT_BULK_ROW_COUNT_RELATION_TAG) {
            int prmIndex;

            prmIndex = 1;
            for (int i = 0; i < INSERT_BULK_ROW_COUNT_RELATION_TAG; i++) {
                populateEntityTagParameters(bulkRelationTagStatement, prmIndex, relationTagBuffer.remove(0));
                prmIndex += INSERT_PRM_COUNT_RELATION_TAG;
            }

            try {
                bulkRelationTagStatement.executeUpdate();
            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to bulk insert relation tags into the database.", e);
            }
        }

        if (complete) {
            while (relationTagBuffer.size() > 0) {
                populateEntityTagParameters(singleRelationTagStatement, 1, relationTagBuffer.remove(0));

                try {
                    singleRelationTagStatement.executeUpdate();
                } catch (SQLException e) {
                    throw new OsmosisRuntimeException("Unable to insert a relation tag into the database.", e);
                }
            }
        }
    }

    /**
     * Flushes relation members to the database. If complete is false, this will only write relation
     * members until the remaining relation member count is less than the multi-row insert statement
     * row count. If complete is true, all remaining rows will be written using single row insert
     * statements.
     *
     * @param complete If true, all data will be written to the database. If false, some data may be
     *        left until more data is available.
     */
    private void flushRelationMembers(boolean complete) {
        while (relationMemberBuffer.size() >= INSERT_BULK_ROW_COUNT_RELATION_MEMBER) {
            int prmIndex;

            prmIndex = 1;
            for (int i = 0; i < INSERT_BULK_ROW_COUNT_RELATION_MEMBER; i++) {
                populateRelationMemberParameters(bulkRelationMemberStatement, prmIndex, relationMemberBuffer.remove(0));
                prmIndex += INSERT_PRM_COUNT_RELATION_MEMBER;
            }

            try {
                bulkRelationMemberStatement.executeUpdate();
            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to bulk insert relation members into the database.", e);
            }
        }

        if (complete) {
            while (relationMemberBuffer.size() > 0) {
                populateRelationMemberParameters(singleRelationMemberStatement, 1, relationMemberBuffer.remove(0));

                try {
                    singleRelationMemberStatement.executeUpdate();
                } catch (SQLException e) {
                    throw new OsmosisRuntimeException("Unable to insert a relation member into the database.", e);
                }
            }
        }
    }
   
   
    private void populateCurrentNodes() {
        // Copy data into the current node tables.
        for (long i = minNodeId; i < maxNodeId; i += LOAD_CURRENT_NODE_ROW_COUNT) {
            // Node
            try {
                loadCurrentNodesStatement.setLong(1, i);
                loadCurrentNodesStatement.setLong(2, i + LOAD_CURRENT_NODE_ROW_COUNT);

                loadCurrentNodesStatement.execute();

            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to load current nodes.", e);
            }

            // Node tags
            try {
                loadCurrentNodeTagsStatement.setLong(1, i);
                loadCurrentNodeTagsStatement.setLong(2, i + LOAD_CURRENT_NODE_ROW_COUNT);

                loadCurrentNodeTagsStatement.execute();

            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to load current node tags.", e);
            }

            dbCtx.commit();
        }
    }
   
   
    private void populateCurrentWays() {
        for (long i = minWayId; i < maxWayId; i += LOAD_CURRENT_WAY_ROW_COUNT) {
            // Way
            try {
                loadCurrentWaysStatement.setLong(1, i);
                loadCurrentWaysStatement.setLong(2, i + LOAD_CURRENT_WAY_ROW_COUNT);

                loadCurrentWaysStatement.execute();

            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to load current ways.", e);
            }

            // Way tags
            try {
                loadCurrentWayTagsStatement.setLong(1, i);
                loadCurrentWayTagsStatement.setLong(2, i + LOAD_CURRENT_WAY_ROW_COUNT);

                loadCurrentWayTagsStatement.execute();

            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to load current way tags.", e);
            }

            // Way nodes
            try {
                loadCurrentWayNodesStatement.setLong(1, i);
                loadCurrentWayNodesStatement.setLong(2, i + LOAD_CURRENT_WAY_ROW_COUNT);

                loadCurrentWayNodesStatement.execute();

            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to load current way nodes.", e);
            }

            dbCtx.commit();
        }
    }
   
   
    private void populateCurrentRelations() {
        for (long i = minRelationId; i < maxRelationId; i += LOAD_CURRENT_RELATION_ROW_COUNT) {
            // Way
            try {
                loadCurrentRelationsStatement.setLong(1, i);
                loadCurrentRelationsStatement.setLong(2, i + LOAD_CURRENT_RELATION_ROW_COUNT);

                loadCurrentRelationsStatement.execute();

            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to load current relations.", e);
            }

            // Relation tags
            try {
                loadCurrentRelationTagsStatement.setLong(1, i);
                loadCurrentRelationTagsStatement.setLong(2, i + LOAD_CURRENT_RELATION_ROW_COUNT);

                loadCurrentRelationTagsStatement.execute();

            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to load current relation tags.", e);
            }

            // Relation members
            try {
                loadCurrentRelationMembersStatement.setLong(1, i);
                loadCurrentRelationMembersStatement.setLong(2, i + LOAD_CURRENT_RELATION_ROW_COUNT);

                loadCurrentRelationMembersStatement.execute();

            } catch (SQLException e) {
                throw new OsmosisRuntimeException("Unable to load current relation members.", e);
            }

            dbCtx.commit();
        }
    }
   
   
    private void populateCurrentTables() {
      if (populateCurrentTables) {
        populateCurrentNodes();
        populateCurrentWays();
        populateCurrentRelations();
        }
    }


    /**
     * {@inheritDoc}
     */
    public void initialize(Map<String, Object> metaData) {
    // Do nothing.
  }


    /**
     * Writes any buffered data to the database and commits.
     */
    @Override
    public void complete() {
        initialize();

        flushNodes(true);
        flushNodeTags(true);
        flushWays(true);
        flushWayTags(true);
        flushWayNodes(true);
        flushRelations(true);
        flushRelationTags(true);
        flushRelationMembers(true);

        // Re-enable indexes now that the load has completed.
        dbCtx.enableIndexes(DISABLE_KEY_TABLES);

        populateCurrentTables();

        // Unlock tables (if they were locked) now that we have completed.
        if (lockTables) {
          dbCtx.unlockTables(LOCK_TABLES);
        }

        dbCtx.commit();
    }

    /**
     * Releases all database resources.
     */
    public void release() {
        userManager.release();

        dbCtx.release();
    }

    /**
     * {@inheritDoc}
     */
    public void process(EntityContainer entityContainer) {
      Entity entity;
     
        initialize();

        entity = entityContainer.getEntity();
        userManager.addOrUpdateUser(entityContainer.getEntity().getUser());
        changesetManager.addChangesetIfRequired(entity.getChangesetId(), entity.getUser());

        entityContainer.process(this);
    }

    /**
     * {@inheritDoc}
     */
    public void process(BoundContainer boundContainer) {
        // Do nothing.
    }

    /**
     * {@inheritDoc}
     */
    public void process(NodeContainer nodeContainer) {
        Node node;
        long nodeId;

        node = nodeContainer.getEntity();
        nodeId = node.getId();

        if (nodeId >= maxNodeId) {
            maxNodeId = nodeId + 1;
        }
        if (nodeId < minNodeId) {
          minNodeId = nodeId;
        }

        nodeBuffer.add(node);

        flushNodes(false);
    }

    /**
     * Process the node tags.
     *
     * @param node The node to be processed.
     */
    private void addNodeTags(Node node) {
        for (Tag tag : node.getTags()) {
            nodeTagBuffer.add(new DbFeatureHistory<DbFeature<Tag>>(new DbFeature<Tag>(node.getId(), tag), node
                    .getVersion()));
        }

        flushNodeTags(false);
    }

    /**
     * {@inheritDoc}
     */
    public void process(WayContainer wayContainer) {
        Way way;
        long wayId;

        flushNodes(true);

        way = wayContainer.getEntity();
        wayId = way.getId();

        if (wayId >= maxWayId) {
            maxWayId = wayId + 1;
        }
        if (wayId < minWayId) {
          minWayId = wayId;
        }
        wayBuffer.add(way);

        flushWays(false);
    }

    /**
     * Process the way tags.
     *
     * @param way The way to be processed.
     */
    private void addWayTags(Way way) {
        for (Tag tag : way.getTags()) {
            wayTagBuffer.add(new DbFeatureHistory<DbFeature<Tag>>(new DbFeature<Tag>(way.getId(), tag), way
                    .getVersion()));
        }

        flushWayTags(false);
    }

    /**
     * Process the way nodes.
     *
     * @param way The way to be processed.
     */
    private void addWayNodes(Way way) {
        List<WayNode> nodeReferenceList;

        nodeReferenceList = way.getWayNodes();

        for (int i = 0; i < nodeReferenceList.size(); i++) {
            wayNodeBuffer.add(new DbFeatureHistory<DbOrderedFeature<WayNode>>(new DbOrderedFeature<WayNode>(
                    way.getId(), nodeReferenceList.get(i), i + 1), way.getVersion()));
        }

        flushWayNodes(false);
    }

    /**
     * {@inheritDoc}
     */
    public void process(RelationContainer relationContainer) {
        Relation relation;
        long relationId;

        flushWays(true);

        relation = relationContainer.getEntity();
        relationId = relation.getId();

        if (relationId >= maxRelationId) {
            maxRelationId = relationId + 1;
        }
        if (relationId < minRelationId) {
          minRelationId = relationId;
        }
        relationBuffer.add(relation);

        flushRelations(false);
    }

    /**
     * Process the relation tags.
     *
     * @param relation The relation to be processed.
     */
    private void addRelationTags(Relation relation) {
        for (Tag tag : relation.getTags()) {
            relationTagBuffer.add(new DbFeatureHistory<DbFeature<Tag>>(new DbFeature<Tag>(relation.getId(), tag),
                    relation.getVersion()));
        }

        flushRelationTags(false);
    }

    /**
     * Process the relation members.
     *
     * @param relation The relation to be processed.
     */
    private void addRelationMembers(Relation relation) {
        List<RelationMember> memberReferenceList;

        memberReferenceList = relation.getMembers();

        for (int i = 0; i < memberReferenceList.size(); i++) {
            relationMemberBuffer.add(new DbFeatureHistory<DbOrderedFeature<RelationMember>>(
                    new DbOrderedFeature<RelationMember>(relation.getId(), memberReferenceList.get(i), i + 1), relation
                            .getVersion()));
        }

        flushRelationMembers(false);
    }
}
TOP

Related Classes of org.openstreetmap.osmosis.apidb.v0_6.ApidbWriter

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.