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

opennlp.tools.namefind.NameFinderEventStream Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 opennlp.tools.namefind;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Objects;

import opennlp.tools.ml.model.Event;
import opennlp.tools.util.ObjectStream;
import opennlp.tools.util.SequenceCodec;
import opennlp.tools.util.Span;
import opennlp.tools.util.featuregen.AdditionalContextFeatureGenerator;
import opennlp.tools.util.featuregen.WindowFeatureGenerator;

/**
 * Class for creating an event stream out of data files for training an {@link TokenNameFinder}.
 */
public class NameFinderEventStream extends opennlp.tools.util.AbstractEventStream {

  private final NameContextGenerator contextGenerator;

  private final AdditionalContextFeatureGenerator additionalContextFeatureGenerator =
      new AdditionalContextFeatureGenerator();

  private final SequenceCodec codec;

  private final String defaultType;

  /**
   * Initializes a {@link NameFinderEventStream} using the specified {@code dataStream} and
   * {@link NameContextGenerator}.
   *
   * @param dataStream The {@link ObjectStream data stream} of events.
   * @param type {@code null} or overrides the type parameter in the provided samples.
   * @param contextGenerator The {@link NameContextGenerator} used to generate features for the event stream.
   * @param codec The {@link SequenceCodec} to use.
   */
  public NameFinderEventStream(ObjectStream dataStream, String type,
                               NameContextGenerator contextGenerator, SequenceCodec codec) {
    super(dataStream);

    this.codec = Objects.requireNonNullElseGet(codec, BioCodec::new);
    this.contextGenerator = contextGenerator;
    this.contextGenerator.addFeatureGenerator(
        new WindowFeatureGenerator(additionalContextFeatureGenerator, 8, 8));

    this.defaultType = type;
  }

  /**
   * Generates the name tag outcomes ({@code start}, {@code continue}, {@code other}) for each
   * token in a sentence with the specified {@code length} using the specified {@link Span names}.
   * 
   * @param names Token {@link Span spans} for each of the names.
   * @param type {@code null} or overrides the type parameter in the provided samples
   * @param length The length of the sentence.
   *               
   * @return An array of {@code start}, {@code continue}, {@code other} outcomes based on the
   *         specified names and sentence {@code length}.
   *
   * @deprecated use the {@link BioCodec} implementation of the SequenceValidator instead!
   */
  @Deprecated(forRemoval = true)
  public static String[] generateOutcomes(Span[] names, String type, int length) {
    String[] outcomes = new String[length];
    Arrays.fill(outcomes, NameFinderME.OTHER);
    for (Span name : names) {
      if (name.getType() == null) {
        outcomes[name.getStart()] = type + "-" + NameFinderME.START;
      }
      else {
        outcomes[name.getStart()] = name.getType() + "-" + NameFinderME.START;
      }
      // now iterate from begin + 1 till end
      for (int i = name.getStart() + 1; i < name.getEnd(); i++) {
        if (name.getType() == null) {
          outcomes[i] = type + "-" + NameFinderME.CONTINUE;
        }
        else {
          outcomes[i] = name.getType() + "-" + NameFinderME.CONTINUE;
        }
      }
    }
    return outcomes;
  }

  /**
   * Generates {@link Event events} for each token in a {@code sentence}
   * with the specified {@code outcomes} using the specified {@link NameContextGenerator}.
   *
   * @param sentence Token representing a sentence.
   * @param outcomes An array of outcomes.
   * @param cg The {@link NameContextGenerator} to use.
   *           
   * @return A list of {@link Event events} generated.
   */
  public static List generateEvents(String[] sentence, String[] outcomes,
                                           NameContextGenerator cg) {
    List events = new ArrayList<>(outcomes.length);
    for (int i = 0; i < outcomes.length; i++) {
      events.add(new Event(outcomes[i], cg.getContext(i, sentence, outcomes,null)));
    }

    cg.updateAdaptiveData(sentence, outcomes);

    return events;
  }

  @Override
  protected Iterator createEvents(NameSample sample) {

    if (sample.isClearAdaptiveDataSet()) {
      contextGenerator.clearAdaptiveData();
    }

    Span[] names = sample.getNames();
    if (!Objects.isNull(this.defaultType)) {
      overrideType(names);
    }

    String[] outcomes = codec.encode(names, sample.getSentence().length);
    // String outcomes[] = generateOutcomes(sample.getNames(), type, sample.getSentence().length);
    additionalContextFeatureGenerator.setCurrentContext(sample.getAdditionalContext());
    String[] tokens = new String[sample.getSentence().length];

    for (int i = 0; i < sample.getSentence().length; i++) {
      tokens[i] = sample.getSentence()[i];
    }

    return generateEvents(tokens, outcomes, contextGenerator).iterator();
  }

  private void overrideType(Span[] names) {
    for (int i = 0; i < names.length; i++) {
      Span n = names[i];
      names[i] = new Span(n.getStart(), n.getEnd(), this.defaultType,
              n.getProb());
    }
  }

  /**
   * Generated previous decision features for each token based on contents of the
   * specified {@code prevMap}.
   *
   * @param tokens The token for which the context is generated.
   * @param prevMap A mapping of tokens to their previous decisions.
   * @return A 2-dimensional array with additional context with features for each token.
   */
  public static String[][] additionalContext(String[] tokens, Map prevMap) {
    String[][] ac = new String[tokens.length][1];
    for (int ti = 0; ti < tokens.length; ti++) {
      String pt = prevMap.get(tokens[ti]);
      ac[ti][0] = "pd=" + pt;
    }
    return ac;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy