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

com.graphaware.tx.event.improved.data.lazy.LazyEntityTransactionData Maven / Gradle / Ivy

There is a newer version: 4.2.0.58
Show newest version
/*
 * Copyright (c) 2013-2018 GraphAware
 *
 * This file is part of the GraphAware Framework.
 *
 * GraphAware Framework is free software: you can redistribute it and/or modify it under
 * the terms of the GNU 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 General Public License for more details. You should have received a copy of
 * the GNU General Public License along with this program.  If not, see
 * .
 */

package com.graphaware.tx.event.improved.data.lazy;


import com.graphaware.common.util.Change;
import com.graphaware.tx.event.improved.data.EntityTransactionData;
import org.neo4j.graphdb.Entity;
import org.neo4j.graphdb.event.PropertyEntry;
import org.neo4j.logging.Log;
import com.graphaware.common.log.LoggerFactory;

import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;

/**
 * {@link com.graphaware.tx.event.improved.data.EntityTransactionData} that lazily initializes its internal structures (indexed transaction data)
 * as they are needed by callers to prevent unnecessary overheads.
 *
 * @param  type of the entity.
 */
public abstract class LazyEntityTransactionData implements EntityTransactionData {
    private static final Log LOG = LoggerFactory.getLogger(LazyEntityTransactionData.class);

    private Map created = null;
    private Map deleted = null;
    private Map> changed = null;

    /**
     * >
     */
    private Map> createdProperties = null;
    /**
     * >
     */
    private Map> deletedProperties = null;
    /**
     * >
     */
    private Map>> changedProperties = null;
    /**
     * > of properties of deleted entities
     */
    private Map> deletedEntityProperties = null;

    /**
     * Create an old snapshot of an original entity.
     *
     * @param original to create a snapshot from.
     * @return the snapshot.
     */
    protected abstract T oldSnapshot(T original);

    /**
     * Create a new snapshot of an original entity.
     *
     * @param original to create a snapshot from.
     * @return the snapshot.
     */
    protected abstract T newSnapshot(T original);

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean hasBeenCreated(T entity) {
        initializeCreated();
        return created.containsKey(entity.getId());
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Collection getAllCreated() {
        initializeCreated();
        return Collections.unmodifiableCollection(created.values());
    }

    private void initializeCreated() {
        if (created == null) {

            created = new HashMap<>();

            for (T created : created()) {
                this.created.put(created.getId(), newSnapshot(created));
            }
        }
    }

    /**
     * Get all entities created in the transaction from the Neo4j API.
     *
     * @return created entities.
     */
    protected abstract Iterable created();

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean hasBeenDeleted(T entity) {
        initializeDeleted();
        return deleted.containsKey(entity.getId());
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public T getDeleted(T entity) {
        initializeDeleted();

        if (!hasBeenDeleted(entity)) {
            throw new IllegalArgumentException(entity + " has not been deleted!");
        }

        return deleted.get(entity.getId());
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Collection getAllDeleted() {
        initializeDeleted();
        return Collections.unmodifiableCollection(deleted.values());
    }

    private void initializeDeleted() {
        if (deleted == null) {

            deleted = new HashMap<>();

            for (T deleted : deleted()) {
                this.deleted.put(deleted.getId(), oldSnapshot(deleted));
            }
        }
    }

    /**
     * Get all entities deleted in the transaction from the Neo4j API.
     *
     * @return created entities.
     */
    protected abstract Iterable deleted();

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean hasBeenChanged(T entity) {
        initializeChanged();
        return changedContainsKey(entity);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Change getChanged(T entity) {
        initializeChanged();

        if (!hasBeenChanged(entity)) {
            throw new IllegalArgumentException(entity + " has not been changed!");
        }

        return changed.get(entity.getId());
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Collection> getAllChanged() {
        initializeChanged();
        return Collections.unmodifiableCollection(changed.values());
    }

    protected void initializeChanged() {
        initializeCreated();
        initializeDeleted();

        if (changed == null) {
            changed = new HashMap<>();

            for (PropertyEntry propertyEntry : assignedProperties()) {
                if (hasNotActuallyChanged(propertyEntry)) {
                    continue;
                }

                T candidate = propertyEntry.entity();
                if (!hasBeenCreated(candidate)) {
                    registerChange(candidate);
                }
            }

            for (PropertyEntry propertyEntry : removedProperties()) {
                T candidate = propertyEntry.entity();
                if (!hasBeenDeleted(candidate)) {
                    registerChange(candidate);
                }
            }

            doInitializeChanged();
        }
    }

    protected void doInitializeChanged() {
        //for subclasses
    }

    protected void registerChange(T candidate) {
        if (!changedContainsKey(candidate)) {
            Change change = createChangeObject(candidate);
            changed.put(candidate.getId(), change);
        }
    }

    protected boolean changedContainsKey(T candidate) {
        return changed.containsKey(candidate.getId());
    }

    protected Change createChangeObject(T candidate) {
        return new Change<>(oldSnapshot(candidate), newSnapshot(candidate));
    }

    /**
     * Get all assigned properties from the Neo4j API.
     *
     * @return assigned properties.
     */
    protected abstract Iterable> assignedProperties();

    /**
     * Get all removed properties from the Neo4j API.
     *
     * @return removed properties.
     */
    protected abstract Iterable> removedProperties();

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean hasPropertyBeenCreated(T entity, String key) {
        initializeProperties();

        if (!hasBeenChanged(entity)) {
            LOG.warn(entity + " has not been changed but the caller thinks it should have created properties.");
            return false;
        }

        if (!createdProperties.containsKey(entity.getId())) {
            return false;
        }

        return createdProperties.get(entity.getId()).containsKey(key);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Map createdProperties(T entity) {
        initializeProperties();

        if (!hasBeenChanged(entity)) {
            LOG.warn(entity + " has not been changed but the caller thinks it should have created properties.");
            return Collections.emptyMap();
        }

        if (!createdProperties.containsKey(entity.getId())) {
            return Collections.emptyMap();
        }

        return Collections.unmodifiableMap(createdProperties.get(entity.getId()));
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean hasPropertyBeenDeleted(T entity, String key) {
        initializeProperties();

        if (!hasBeenChanged(entity)) {
            LOG.warn(entity + " has not been changed but the caller thinks it should have deleted properties.");
            return false;
        }

        if (!deletedProperties.containsKey(entity.getId())) {
            return false;
        }

        return deletedProperties.get(entity.getId()).containsKey(key);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Map deletedProperties(T entity) {
        initializeProperties();

        if (!hasBeenChanged(entity)) {
            LOG.warn(entity + " has not been changed but the caller thinks it should have deleted properties.");
            return Collections.emptyMap();
        }

        if (!deletedProperties.containsKey(entity.getId())) {
            return Collections.emptyMap();
        }

        return Collections.unmodifiableMap(deletedProperties.get(entity.getId()));
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Map propertiesOfDeletedEntity(T entity) {
        initializeProperties();

        if (!hasBeenDeleted(entity)) {
            LOG.error(entity + " has not been deleted but the caller thinks it has! This is a bug.");
            throw new IllegalStateException(entity + " has not been deleted but the caller thinks it has! This is a bug.");
        }

        if (!deletedEntityProperties.containsKey(entity.getId())) {
            return Collections.emptyMap();
        }

        return Collections.unmodifiableMap(deletedEntityProperties.get(entity.getId()));
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean hasPropertyBeenChanged(T entity, String key) {
        initializeProperties();

        if (!hasBeenChanged(entity)) {
            LOG.warn(entity + " has not been changed but the caller thinks it should have changed properties.");
            return false;
        }

        if (!changedProperties.containsKey(entity.getId())) {
            return false;
        }

        return changedProperties.get(entity.getId()).containsKey(key);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Map> changedProperties(T entity) {
        initializeProperties();

        if (!hasBeenChanged(entity)) {
            LOG.warn(entity + " has not been changed but the caller thinks it should have changed properties.");
            return Collections.emptyMap();
        }

        if (!changedProperties.containsKey(entity.getId())) {
            return Collections.emptyMap();
        }

        return Collections.unmodifiableMap(changedProperties.get(entity.getId()));
    }

    private void initializeProperties() {
        if (createdProperties != null) {
            assert changedProperties != null;
            assert deletedProperties != null;
            assert deletedEntityProperties != null;

            return;
        }

        //initializeCreated(); // - called by initializeChanged()
        //initializeDeleted(); // - called by initializeChanged()
        initializeChanged();

        createdProperties = new HashMap<>();
        deletedProperties = new HashMap<>();
        changedProperties = new HashMap<>();
        deletedEntityProperties = new HashMap<>();

        for (PropertyEntry propertyEntry : assignedProperties()) {
            T entity = propertyEntry.entity();

            if (hasBeenCreated(entity)) {
                continue;
            }

            if (hasNotActuallyChanged(propertyEntry)) {
                continue;
            }

            if (propertyEntry.previouslyCommitedValue() == null) {
                if (!createdProperties.containsKey(entity.getId())) {
                    createdProperties.put(entity.getId(), new HashMap());
                }
                createdProperties.get(entity.getId()).put(propertyEntry.key(), propertyEntry.value());
            } else {
                if (!changedProperties.containsKey(entity.getId())) {
                    changedProperties.put(entity.getId(), new HashMap>());
                }
                changedProperties.get(entity.getId()).put(propertyEntry.key(), new Change<>(propertyEntry.previouslyCommitedValue(), propertyEntry.value()));
            }
        }

        for (PropertyEntry propertyEntry : removedProperties()) {
            T entity = propertyEntry.entity();

            if (deleted.containsKey(entity.getId())) {
                if (!deletedEntityProperties.containsKey(entity.getId())) {
                    deletedEntityProperties.put(entity.getId(), new HashMap());
                }
                deletedEntityProperties.get(entity.getId()).put(propertyEntry.key(), propertyEntry.previouslyCommitedValue());
                continue;
            }

            if (!changedContainsKey(entity)) {
                throw new IllegalStateException(entity + " seems to have not been deleted or changed, this is a bug");
            }

            assert changedContainsKey(entity);

            if (!deletedProperties.containsKey(entity.getId())) {
                deletedProperties.put(entity.getId(), new HashMap());
            }

            deletedProperties.get(entity.getId()).put(propertyEntry.key(), propertyEntry.previouslyCommitedValue());
        }
    }

    private boolean hasNotActuallyChanged(PropertyEntry propertyEntry) {
        return propertyEntry.previouslyCommitedValue() != null && propertyEntry.previouslyCommitedValue().equals(propertyEntry.value());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy