com.b2international.snowowl.snomed.datastore.index.change.RelationshipChangeProcessor.java Source code

Java tutorial

Introduction

Here is the source code for com.b2international.snowowl.snomed.datastore.index.change.RelationshipChangeProcessor.java

Source

/*
 * Copyright 2011-2016 B2i Healthcare Pte Ltd, http://b2i.sg
 * 
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.b2international.snowowl.snomed.datastore.index.change;

import static com.google.common.collect.Sets.newHashSet;

import java.io.IOException;
import java.util.Collection;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;

import com.b2international.index.Hits;
import com.b2international.index.query.Query;
import com.b2international.index.revision.RevisionSearcher;
import com.b2international.snowowl.core.api.ComponentUtils;
import com.b2international.snowowl.datastore.ICDOCommitChangeSet;
import com.b2international.snowowl.datastore.index.ChangeSetProcessorBase;
import com.b2international.snowowl.snomed.Relationship;
import com.b2international.snowowl.snomed.SnomedPackage;
import com.b2international.snowowl.snomed.common.SnomedTerminologyComponentConstants;
import com.b2international.snowowl.snomed.datastore.index.entry.SnomedRelationshipIndexEntry;
import com.b2international.snowowl.snomed.datastore.index.entry.SnomedRelationshipIndexEntry.Builder;
import com.b2international.snowowl.snomed.datastore.index.refset.RefSetMemberChange;
import com.b2international.snowowl.snomed.datastore.index.update.ReferenceSetMembershipUpdater;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Maps;
import com.google.common.collect.Multimap;

/**
 * @since 4.3
 */
public class RelationshipChangeProcessor extends ChangeSetProcessorBase {

    private final ReferringMemberChangeProcessor memberChangeProcessor;

    public RelationshipChangeProcessor() {
        super("relationship changes");
        this.memberChangeProcessor = new ReferringMemberChangeProcessor(
                SnomedTerminologyComponentConstants.RELATIONSHIP_NUMBER);
    }

    @Override
    public void process(ICDOCommitChangeSet commitChangeSet, RevisionSearcher searcher) throws IOException {
        final Multimap<String, RefSetMemberChange> referringRefSets = memberChangeProcessor.process(commitChangeSet,
                searcher);

        deleteRevisions(SnomedRelationshipIndexEntry.class,
                commitChangeSet.getDetachedComponents(SnomedPackage.Literals.RELATIONSHIP));

        final Map<String, Relationship> newRelationshipsById = StreamSupport
                .stream(commitChangeSet.getNewComponents(Relationship.class).spliterator(), false)
                .collect(Collectors.toMap(relationship -> relationship.getId(), relationship -> relationship));

        for (Relationship relationship : commitChangeSet.getNewComponents(Relationship.class)) {
            final Builder doc = SnomedRelationshipIndexEntry.builder(relationship);
            indexNewRevision(relationship.cdoID(), doc.build());
        }

        final Map<String, Relationship> changedRelationshipsById = StreamSupport
                .stream(commitChangeSet.getDirtyComponents(Relationship.class).spliterator(), false)
                .collect(Collectors.toMap(relationship -> relationship.getId(), relationship -> relationship));

        final Set<String> changedRelationshipIds = newHashSet(changedRelationshipsById.keySet());
        final Set<String> referencedRelationshipIds = newHashSet(referringRefSets.keySet());
        referencedRelationshipIds.removeAll(newRelationshipsById.keySet());
        changedRelationshipIds.addAll(referencedRelationshipIds);

        final Query<SnomedRelationshipIndexEntry> query = Query.select(SnomedRelationshipIndexEntry.class)
                .where(SnomedRelationshipIndexEntry.Expressions.ids(changedRelationshipIds))
                .limit(changedRelationshipIds.size()).build();

        final Hits<SnomedRelationshipIndexEntry> changedRelationshipHits = searcher.search(query);
        final ImmutableMap<String, SnomedRelationshipIndexEntry> changedRelationshipRevisionsById = Maps
                .uniqueIndex(changedRelationshipHits, ComponentUtils.<String>getIdFunction());

        for (final String id : changedRelationshipIds) {
            final SnomedRelationshipIndexEntry currentDoc = changedRelationshipRevisionsById.get(id);
            if (currentDoc == null) {
                throw new IllegalStateException(
                        String.format("Current relationship revision should not be null for %s", id));
            }

            final Relationship relationship = changedRelationshipsById.get(id);
            final Builder doc;
            if (relationship != null) {
                doc = SnomedRelationshipIndexEntry.builder(relationship);
            } else {
                doc = SnomedRelationshipIndexEntry.builder(currentDoc);
            }

            final Collection<String> currentMemberOf = currentDoc.getMemberOf();
            final Collection<String> currentActiveMemberOf = currentDoc.getActiveMemberOf();
            new ReferenceSetMembershipUpdater(referringRefSets.removeAll(id), currentMemberOf,
                    currentActiveMemberOf).update(doc);

            indexChangedRevision(currentDoc.getStorageKey(), doc.build());
        }
    }

}