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

Java tutorial

Introduction

Here is the source code for com.b2international.snowowl.snomed.datastore.index.change.DescriptionChangeProcessor.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.Collections;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;

import org.eclipse.emf.cdo.common.id.CDOIDUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.b2international.index.Hits;
import com.b2international.index.query.Query;
import com.b2international.index.revision.RevisionSearcher;
import com.b2international.snowowl.datastore.ICDOCommitChangeSet;
import com.b2international.snowowl.datastore.index.ChangeSetProcessorBase;
import com.b2international.snowowl.snomed.Description;
import com.b2international.snowowl.snomed.SnomedPackage;
import com.b2international.snowowl.snomed.common.SnomedTerminologyComponentConstants;
import com.b2international.snowowl.snomed.core.domain.Acceptability;
import com.b2international.snowowl.snomed.datastore.SnomedDatastoreActivator;
import com.b2international.snowowl.snomed.datastore.index.entry.SnomedDescriptionIndexEntry;
import com.b2international.snowowl.snomed.datastore.index.entry.SnomedDescriptionIndexEntry.Builder;
import com.b2international.snowowl.snomed.datastore.index.entry.SnomedDocument;
import com.b2international.snowowl.snomed.datastore.index.refset.RefSetMemberChange;
import com.b2international.snowowl.snomed.datastore.index.update.ReferenceSetMembershipUpdater;
import com.google.common.base.Joiner;
import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableMultimap;
import com.google.common.collect.Iterables;
import com.google.common.collect.Multimap;

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

    private static final Logger LOG = LoggerFactory.getLogger(DescriptionChangeProcessor.class);

    private final ReferringMemberChangeProcessor memberChangeProcessor;

    public DescriptionChangeProcessor() {
        super("description changes");
        this.memberChangeProcessor = new ReferringMemberChangeProcessor(
                SnomedTerminologyComponentConstants.DESCRIPTION_NUMBER);
    }

    @Override
    public void process(ICDOCommitChangeSet commitChangeSet, RevisionSearcher searcher) throws IOException {
        final Map<String, Multimap<Acceptability, RefSetMemberChange>> acceptabilityChangesByDescription = new DescriptionAcceptabilityChangeProcessor()
                .process(commitChangeSet, searcher);

        final Multimap<String, RefSetMemberChange> referringRefSets = HashMultimap
                .create(memberChangeProcessor.process(commitChangeSet, searcher));

        // delete detached descriptions
        deleteRevisions(SnomedDescriptionIndexEntry.class,
                commitChangeSet.getDetachedComponents(SnomedPackage.Literals.DESCRIPTION));

        // (re)index new and dirty descriptions
        final Map<String, Description> newDescriptionsById = StreamSupport
                .stream(commitChangeSet.getNewComponents(Description.class).spliterator(), false)
                .collect(Collectors.toMap(description -> description.getId(), description -> description));

        final Map<String, Description> changedDescriptionsById = StreamSupport
                .stream(commitChangeSet.getDirtyComponents(Description.class).spliterator(), false)
                .collect(Collectors.toMap(description -> description.getId(), description -> description));

        final Set<String> changedDescriptionIds = newHashSet(changedDescriptionsById.keySet());
        final Set<String> referencedDescriptionIds = newHashSet(referringRefSets.keySet());
        referencedDescriptionIds.removeAll(newDescriptionsById.keySet());
        changedDescriptionIds.addAll(referencedDescriptionIds);

        // load the known descriptions 
        final Query<SnomedDescriptionIndexEntry> query = Query.select(SnomedDescriptionIndexEntry.class)
                .where(SnomedDescriptionIndexEntry.Expressions.ids(changedDescriptionIds)).limit(Integer.MAX_VALUE)
                .build();

        final Hits<SnomedDescriptionIndexEntry> changedDescriptionHits = searcher.search(query);

        Multimap<String, SnomedDescriptionIndexEntry> changedDescriptionRevisionsById = ArrayListMultimap
                .<String, SnomedDescriptionIndexEntry>create();
        changedDescriptionHits.forEach(hit -> changedDescriptionRevisionsById.put(hit.getId(), hit));

        boolean reindexRunning = isReindexRunning(SnomedDatastoreActivator.REPOSITORY_UUID);

        for (Entry<String, Collection<SnomedDescriptionIndexEntry>> entry : changedDescriptionRevisionsById.asMap()
                .entrySet()) {
            if (entry.getValue().size() > 1) {

                String message = String.format("Description with id %s exists with multiple storage keys: [%s]",
                        entry.getKey(), Joiner.on(", ").join(entry.getValue().stream().map(e -> e.getStorageKey())
                                .collect(Collectors.toList())));

                if (reindexRunning) {
                    LOG.info(message);
                } else {
                    throw new IllegalStateException(message);
                }
            }
        }

        // load missing descriptions with only changed acceptability values
        final Set<String> descriptionsToBeLoaded = newHashSet();
        for (String descriptionWithAccepatibilityChange : acceptabilityChangesByDescription.keySet()) {
            if (!newDescriptionsById.containsKey(descriptionWithAccepatibilityChange)
                    && !changedDescriptionIds.contains(descriptionWithAccepatibilityChange)) {
                descriptionsToBeLoaded.add(descriptionWithAccepatibilityChange);
            }
        }

        // process changes
        for (final String id : Iterables.concat(newDescriptionsById.keySet(), changedDescriptionIds)) {
            if (newDescriptionsById.containsKey(id)) {
                final Description description = newDescriptionsById.get(id);
                final long storageKey = CDOIDUtil.getLong(description.cdoID());
                final Builder doc = SnomedDescriptionIndexEntry.builder(description);
                processChanges(id, doc, null, acceptabilityChangesByDescription.get(id), referringRefSets);
                indexNewRevision(storageKey, doc.build());
            } else if (changedDescriptionIds.contains(id)) {
                final Collection<SnomedDescriptionIndexEntry> docs = changedDescriptionRevisionsById.get(id);
                for (SnomedDescriptionIndexEntry currentDoc : docs) {

                    if (currentDoc == null) {
                        throw new IllegalStateException(
                                String.format("Current description revision should not be null for: %s", id));
                    }

                    final Description description = changedDescriptionsById.get(id);
                    final Builder doc;
                    if (description != null) {
                        doc = SnomedDescriptionIndexEntry.builder(description);
                    } else {
                        doc = SnomedDescriptionIndexEntry.builder(currentDoc);
                    }

                    processChanges(id, doc, currentDoc, acceptabilityChangesByDescription.get(id),
                            referringRefSets);
                    indexChangedRevision(currentDoc.getStorageKey(), doc.build());

                }

            } else {
                throw new IllegalStateException(
                        String.format("Description %s is missing from new and dirty maps", id));
            }
        }

        // process cascading acceptability changes in unchanged docs
        if (!descriptionsToBeLoaded.isEmpty()) {
            final Query<SnomedDescriptionIndexEntry> descriptionsToBeLoadedQuery = Query
                    .select(SnomedDescriptionIndexEntry.class)
                    .where(SnomedDocument.Expressions.ids(descriptionsToBeLoaded))
                    .limit(descriptionsToBeLoaded.size()).build();

            for (SnomedDescriptionIndexEntry unchangedDescription : searcher.search(descriptionsToBeLoadedQuery)) {
                final Builder doc = SnomedDescriptionIndexEntry.builder(unchangedDescription);
                processChanges(unchangedDescription.getId(), doc, unchangedDescription,
                        acceptabilityChangesByDescription.get(unchangedDescription.getId()),
                        HashMultimap.<String, RefSetMemberChange>create());
                indexChangedRevision(unchangedDescription.getStorageKey(), doc.build());
            }
        }
    }

    private void processChanges(final String id, final Builder doc,
            final SnomedDescriptionIndexEntry currentRevision,
            Multimap<Acceptability, RefSetMemberChange> acceptabilityChanges,
            Multimap<String, RefSetMemberChange> referringRefSets) {
        final Multimap<Acceptability, String> acceptabilityMap = currentRevision == null
                ? ImmutableMultimap.<Acceptability, String>of()
                : ImmutableMap.copyOf(currentRevision.getAcceptabilityMap()).asMultimap().inverse();

        final Collection<String> preferredLanguageRefSets = newHashSet(
                acceptabilityMap.get(Acceptability.PREFERRED));
        final Collection<String> acceptableLanguageRefSets = newHashSet(
                acceptabilityMap.get(Acceptability.ACCEPTABLE));

        if (acceptabilityChanges != null) {
            collectChanges(acceptabilityChanges.get(Acceptability.PREFERRED), preferredLanguageRefSets);
            collectChanges(acceptabilityChanges.get(Acceptability.ACCEPTABLE), acceptableLanguageRefSets);
        }

        for (String preferredLanguageRefSet : preferredLanguageRefSets) {
            doc.acceptability(preferredLanguageRefSet, Acceptability.PREFERRED);
        }

        for (String acceptableLanguageRefSet : acceptableLanguageRefSets) {
            doc.acceptability(acceptableLanguageRefSet, Acceptability.ACCEPTABLE);
        }

        final Collection<String> currentMemberOf = currentRevision == null ? Collections.<String>emptySet()
                : currentRevision.getMemberOf();
        final Collection<String> currentActiveMemberOf = currentRevision == null ? Collections.<String>emptySet()
                : currentRevision.getActiveMemberOf();
        new ReferenceSetMembershipUpdater(referringRefSets.removeAll(id), currentMemberOf, currentActiveMemberOf)
                .update(doc);
    }

    private void collectChanges(Collection<RefSetMemberChange> changes, Collection<String> refSetIds) {
        for (final RefSetMemberChange change : changes) {
            switch (change.getChangeKind()) {
            case REMOVED:
                refSetIds.remove(change.getRefSetId());
                break;
            default:
                break;
            }
        }

        for (final RefSetMemberChange change : changes) {
            switch (change.getChangeKind()) {
            case ADDED:
                refSetIds.add(change.getRefSetId());
                break;
            default:
                break;
            }
        }
    }

}