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

opennlp.tools.tokenize.DetokenizationDictionary Maven / Gradle / Ivy

There is a newer version: 2.5.0
Show 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.tokenize;

import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.file.Path;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import opennlp.tools.dictionary.serializer.Attributes;
import opennlp.tools.dictionary.serializer.DictionaryEntryPersistor;
import opennlp.tools.dictionary.serializer.Entry;
import opennlp.tools.util.InvalidFormatException;
import opennlp.tools.util.StringList;

public class DetokenizationDictionary {

  public enum Operation {

    /**
     * Attaches the token to the token on the right side.
     */
    MOVE_RIGHT,

    /**
     * Attaches the token to the token on the left side.
     */
    MOVE_LEFT,

    /**
     * Attaches the token to the token on the left and right sides.
     */
    MOVE_BOTH,

    /**
     * Attaches the token to the right token on first occurrence, and
     * to the token on the left side on the second occurrence.
     */
    RIGHT_LEFT_MATCHING;

    /**
     * @param operation The string representation for which an {@link Operation}
     *                  instance is to be found.
     * @return The {@link Operation enum} instance that matches the given {@code operation},
     *         or {@code null} if the input has no equivalent.
     */
    public static Operation parse(String operation) {
      if (operation == null) {
        return null;
      } else {
        if (MOVE_RIGHT.toString().equals(operation)) {
          return MOVE_RIGHT;
        }
        else if (MOVE_LEFT.toString().equals(operation)) {
          return MOVE_LEFT;
        }
        else if (MOVE_BOTH.toString().equals(operation)) {
          return MOVE_BOTH;
        }
        else if (RIGHT_LEFT_MATCHING.toString().equals(operation)) {
          return RIGHT_LEFT_MATCHING;
        }
        else {
          return null;
        }
      }
    }
  }

  private final Map operationTable = new HashMap<>();

  /**
   * Initializes a {@link DetokenizationDictionary} instance.
   *
   * @param tokens An array of tokens that should be de-tokenized according to {@code operations}.
   * @param operations An array of operations which specifies which operation
   *        should be used for the provided {@code tokens}.
   */
  public DetokenizationDictionary(String[] tokens,
      DetokenizationDictionary.Operation[] operations) {
    if (tokens.length != operations.length)
      throw new IllegalArgumentException("tokens and ops must have the same length: tokens=" +
          tokens.length + ", operations=" + operations.length + "!");

    for (int i = 0; i < tokens.length; i++) {
      String token = tokens[i];
      DetokenizationDictionary.Operation operation = operations[i];

      if (token == null)
        throw new IllegalArgumentException("token at index " + i + " must not be null!");

      if (operation == null)
        throw new IllegalArgumentException("operation at index " + i + " must not be null!");

      operationTable.put(token, operation);
    }
  }

  /**
   * Initializes a {@link DetokenizationDictionary} instance via a valid {@link InputStream}.
   *
   * @param in The {@link InputStream} used for loading the dictionary.
   *
   * @throws IOException Thrown if IO errors occurred during initialization.
   */
  public DetokenizationDictionary(InputStream in) throws IOException {
    init(in);
  }

  /**
   * Initializes a {@link DetokenizationDictionary} instance via a valid {@link File}.
   *
   * @param file The {@link File} used for loading the dictionary.
   *
   * @throws IOException Thrown if IO errors occurred during initialization.
   */
  public DetokenizationDictionary(File file) throws IOException {
    try (InputStream in = new BufferedInputStream(new FileInputStream(file))) {
      init(in);
    }
  }

  /**
   * Initializes a {@link DetokenizationDictionary} instance via a valid {@link Path}.
   *
   * @param path The {@link Path} used for loading the dictionary.
   *
   * @throws IOException Thrown if IO errors occurred during initialization.
   */
  public DetokenizationDictionary(Path path) throws IOException {
    this(path.toFile());
  }

  /*
   * Builds up the dictionary from an InputStream.
   */
  private void init(InputStream in) throws IOException {
    DictionaryEntryPersistor.create(in, entry -> {

      String operationString = entry.attributes().getValue("operation");

      StringList word = entry.tokens();

      if (word.size() != 1)
        throw new InvalidFormatException("Each entry must have exactly one token! " + word);

      // parse operation
      Operation operation = Operation.parse(operationString);

      if (operation == null)
        throw new InvalidFormatException("Unknown operation type: " + operationString);

      operationTable.put(word.getToken(0), operation);
    });
  }

  /**
   * @param token The input string for which a valid {@link Operation} is to be found.
   * @return The {@link Operation} that fits the given {@code token}.
   */
  DetokenizationDictionary.Operation getOperation(String token) {
    return operationTable.get(token);
  }

  /**
   * Serializes the current state of a {@link DetokenizationDictionary} via an
   * {@link OutputStream output stream}.
   *
   * @param out A valid, open {@link OutputStream} ready to be used for serialization.
   * @throws IOException  Thrown if IO errors occurred during serialization.
   */
  public void serialize(OutputStream out) throws IOException {
    Iterator entries = new Iterator<>() {

      final Iterator iterator = operationTable.keySet().iterator();

      public boolean hasNext() {
        return iterator.hasNext();
      }

      public Entry next() {

        String token = iterator.next();

        Attributes attributes = new Attributes();
        attributes.setValue("operation", getOperation(token).toString());

        return new Entry(new StringList(token), attributes);
      }

      public void remove() {
        throw new UnsupportedOperationException();
      }
    };

    DictionaryEntryPersistor.serialize(out, entries, false);
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy