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

boofcv.io.recognition.RecognitionIO Maven / Gradle / Ivy

Go to download

BoofCV is an open source Java library for real-time computer vision and robotics applications.

There is a newer version: 1.1.7
Show newest version
/*
 * Copyright (c) 2021, Peter Abeles. All Rights Reserved.
 *
 * This file is part of BoofCV (http://boofcv.org).
 *
 * 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 boofcv.io.recognition;

import boofcv.BoofVersion;
import boofcv.abst.scene.ConfigFeatureToSceneRecognition;
import boofcv.abst.scene.SceneRecognition;
import boofcv.abst.scene.WrapFeatureToSceneRecognition;
import boofcv.abst.scene.ann.FeatureSceneRecognitionNearestNeighbor;
import boofcv.abst.scene.nister2006.FeatureSceneRecognitionNister2006;
import boofcv.alg.scene.ann.RecognitionNearestNeighborInvertedFile;
import boofcv.alg.scene.bow.InvertedFile;
import boofcv.alg.scene.nister2006.RecognitionVocabularyTreeNister2006;
import boofcv.alg.scene.vocabtree.HierarchicalVocabularyTree;
import boofcv.factory.scene.FactorySceneRecognition;
import boofcv.io.UtilIO;
import boofcv.misc.BoofMiscOps;
import boofcv.struct.PackedArray;
import boofcv.struct.feature.*;
import boofcv.struct.image.ImageBase;
import boofcv.struct.image.ImageType;
import boofcv.struct.kmeans.TuplePointDistanceEuclideanSq;
import boofcv.struct.kmeans.TuplePointDistanceHamming;
import deepboof.io.DeepBoofDataBaseOps;
import org.ddogleg.clustering.PointDistance;
import org.ddogleg.struct.BigDogArray_I32;
import org.ddogleg.struct.DogArray;
import org.ddogleg.struct.FastAccess;
import org.jetbrains.annotations.Nullable;

import java.io.*;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;

/**
 * Reading and writing data structures related to recognition.
 *
 * @author Peter Abeles
 **/
@SuppressWarnings({"unchecked", "rawtypes"})
public class RecognitionIO {

	// save file names
	public static final String CONFIG_NAME = "config.yaml";
	public static final String IMAGE_ID_NAME = "image_ids.yaml";
	public static final String DATABASE_NAME = "database.bin";
	public static final String DICTIONARY_NAME = "dictionary.bin";
	public static final String INVERTED_NAME = "inverted_files.bin";

	/**
	 * Downloads then loads the pre-built default scene recognition model. The image DB will of course be empty.
	 * If the decompressed directory already exists then it will skip the download step.
	 *
	 * @param destination Where it should download the model to
	 * @param imageType Type of image the resulting algorithm will process
	 * @return A new instance of {@link SceneRecognition} with an already built model.
	 */
	public static , TD extends TupleDesc>
	WrapFeatureToSceneRecognition downloadDefaultSceneRecognition( File destination,
																			  ImageType imageType ) {
		if (!destination.exists()) {
			BoofMiscOps.checkTrue(destination.mkdirs());
		} else if (destination.isFile())
			throw new IllegalArgumentException("Destination must be a directory not a file");

		File modelDir = new File(destination, "scene_recognition");
		if (!modelDir.exists())
			DeepBoofDataBaseOps.downloadModel(
					"http://boofcv.org/notwiki/largefiles/scene_recognition_default38_inria_holidays.zip", destination);

		return loadFeatureToScene(modelDir, imageType);
	}

	/**
	 * Saves {@link WrapFeatureToSceneRecognition} to disk inside of the specified directory
	 *
	 * @param def What is to be saved
	 * @param dir Direction that it is to be saved
	 */
	public static >
	void saveFeatureToScene( WrapFeatureToSceneRecognition def, File dir ) {
		if (dir.exists() && !dir.isDirectory())
			throw new IllegalArgumentException("Destination must not exist or be a directory");
		if (!dir.exists())
			BoofMiscOps.checkTrue(dir.mkdirs());

		UtilIO.saveConfig(def.getConfig(), new File(dir, CONFIG_NAME));

		List listImageIds = null;

		switch (def.getConfig().typeRecognize) {
			case NISTER_2006 -> {
				FeatureSceneRecognitionNister2006 recognizer = def.getRecognizer();
				saveTreeBin(recognizer.getDatabase(), new File(dir, DATABASE_NAME));
				listImageIds = recognizer.getImageIds();
			}

			case NEAREST_NEIGHBOR -> {
				FeatureSceneRecognitionNearestNeighbor recognizer = def.getRecognizer();

				saveDictionaryBin(recognizer.getDictionary(),
						recognizer.getTupleDOF(),
						recognizer.getDescriptorType(), new File(dir, DICTIONARY_NAME));
				saveNearestNeighborBin(recognizer.getDatabase(), new File(dir, INVERTED_NAME));
				listImageIds = recognizer.getImageIds();
			}
		}
		Objects.requireNonNull(listImageIds);

		if (!listImageIds.isEmpty()) {
			UtilIO.saveListStringYaml(listImageIds, new File(dir, IMAGE_ID_NAME));
		}
	}

	public static , TD extends TupleDesc>
	WrapFeatureToSceneRecognition loadFeatureToScene( File dir, ImageType imageType ) {
		if (!dir.exists())
			throw new IllegalArgumentException("Directory doesn't exist: " + dir.getPath());
		if (!dir.isDirectory())
			throw new IllegalArgumentException("Path is not a directory: " + dir.getPath());

		ConfigFeatureToSceneRecognition config = UtilIO.loadConfig(new File(dir, CONFIG_NAME));
		WrapFeatureToSceneRecognition alg = FactorySceneRecognition.createFeatureToScene(config, imageType);

		// If it's just the model and doesn't have image list then don't try to load it
		boolean loadImagesIDs = new File(dir, IMAGE_ID_NAME).exists();

		switch (config.typeRecognize) {
			case NISTER_2006 -> {
				FeatureSceneRecognitionNister2006 recognizer = alg.getRecognizer();

				loadTreeBin(new File(dir, DATABASE_NAME), recognizer.getDatabase());
				if (loadImagesIDs)
					recognizer.getImageIds().addAll(UtilIO.loadListStringYaml(new File(dir, IMAGE_ID_NAME)));

				// Need to do this so that the tree reference is correctly set up
				recognizer.setDatabase(recognizer.getDatabase());
			}

			case NEAREST_NEIGHBOR -> {
				FeatureSceneRecognitionNearestNeighbor recognizer = alg.getRecognizer();

				// Add the dictionary
				List dictionary = loadDictionaryBin(new File(dir, DICTIONARY_NAME));
				recognizer.setDictionary(dictionary);
				loadNearestNeighborBin(new File(dir, INVERTED_NAME), recognizer.getDatabase());

				// Add the images now
				if (loadImagesIDs) {
					recognizer.getImageIds().addAll(UtilIO.loadListStringYaml(new File(dir, IMAGE_ID_NAME)));
				}
			}

			default -> throw new IllegalArgumentException("Unknown type: " + config.typeRecognize);
		}

		return alg;
	}

	/**
	 * Saves {@link FeatureSceneRecognitionNister2006} to disk inside of the specified directory
	 *
	 * @param def What is to be saved
	 * @param dir Direction that it is to be saved
	 */
	public static >
	void saveNister2006( FeatureSceneRecognitionNister2006 def, File dir ) {
		if (dir.exists() && !dir.isDirectory())
			throw new IllegalArgumentException("Destination must not exist or be a directory");
		if (!dir.exists())
			BoofMiscOps.checkTrue(dir.mkdirs());

		UtilIO.saveConfig(def.getConfig(), new File(dir, CONFIG_NAME));
		saveTreeBin(def.getDatabase(), new File(dir, DATABASE_NAME));
		UtilIO.saveListStringYaml(def.getImageIds(), new File(dir, IMAGE_ID_NAME));
	}

	/**
	 * Loads {@link FeatureSceneRecognitionNister2006}
	 *
	 * @param dir Directory containing saved graph
	 * @param recognizer (Output) where it's loaded into
	 */
	public static >
	void loadNister2006( File dir, FeatureSceneRecognitionNister2006 recognizer ) {
		if (!dir.exists())
			throw new IllegalArgumentException("Directory doesn't exist: " + dir.getPath());
		if (!dir.isDirectory())
			throw new IllegalArgumentException("Path is not a directory: " + dir.getPath());

		loadTreeBin(new File(dir, DATABASE_NAME), recognizer.getDatabase());
		recognizer.getImageIds().addAll(UtilIO.loadListStringYaml(new File(dir, IMAGE_ID_NAME)));

		// Need to do this so that the tree reference is correctly set up
		recognizer.setDatabase(recognizer.getDatabase());
	}

	public static > void saveBin( HierarchicalVocabularyTree tree, File file ) {
		try (var out = new FileOutputStream(file)) {
			saveTreeBin(tree, out);
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static >
	HierarchicalVocabularyTree loadTreeBin( File file, @Nullable HierarchicalVocabularyTree tree ) {
		try (var in = new FileInputStream(file)) {
			return loadTreeBin(in, tree);
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static > void saveDictionaryBin( List dictionary,
																	 int dof, Class descType, File file ) {
		try (var out = new FileOutputStream(file)) {
			saveDictionaryBin(dictionary, dof, descType, out);
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static >
	List loadDictionaryBin( File file ) {
		try (var in = new FileInputStream(file)) {
			return loadDictionaryBin(in);
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static void saveNearestNeighborBin( RecognitionNearestNeighborInvertedFile nn, File file ) {
		try (var out = new FileOutputStream(file)) {
			saveNearestNeighborBin(nn, out);
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static void loadNearestNeighborBin( File file, RecognitionNearestNeighborInvertedFile nn ) {
		try (var in = new FileInputStream(file)) {
			loadNearestNeighborBin(in, nn);
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	/**
	 * Saves the tree in binary format. The format is described in an ascii header.
	 *
	 * @param tree (Input) Tree that's saved
	 * @param out (Output) Stream the tree is written to
	 */
	public static > void saveTreeBin( HierarchicalVocabularyTree tree, OutputStream out ) {
		String header = "BOOFCV_HIERARCHICAL_VOCABULARY_TREE\n";
		header += "# Graph format: id=int,parent=int,branch=int,descIdx=int,dataIdx=int,weight=double,children.size=int,children=int[]\n";
		header += "# Description format: raw array used internally\n";
		header += "format_version 1\n";
		header += "boofcv_version " + BoofVersion.VERSION + "\n";
		header += "git_sha " + BoofVersion.GIT_SHA + "\n";
		header += "branch_factor " + tree.branchFactor + "\n";
		header += "maximum_level " + tree.maximumLevel + "\n";
		header += "nodes.size " + tree.nodes.size + "\n";
		header += "descriptions.size " + tree.descriptions.size() + "\n";
		header += "point_type " + tree.descriptions.getElementType().getSimpleName() + "\n";
		header += "point_dof " + tree.descriptions.getTemp(0).size() + "\n";
		header += "distance.name " + tree.distanceFunction.getClass().getName() + "\n";
		header += "BEGIN_GRAPH\n";
		try {
			out.write(header.getBytes(StandardCharsets.UTF_8));

			DataOutputStream dout = new DataOutputStream(out);
			for (int nodeIdx = 0; nodeIdx < tree.nodes.size; nodeIdx++) {
				HierarchicalVocabularyTree.Node n = tree.nodes.get(nodeIdx);
				dout.writeInt(n.index);
				dout.writeInt(n.parent);
				dout.writeInt(n.branch);
				dout.writeInt(n.descIdx);
				dout.writeInt(n.userIdx);
				dout.writeDouble(n.weight);
				dout.writeInt(n.childrenIndexes.size);
				for (int i = 0; i < n.childrenIndexes.size; i++) {
					dout.writeInt(n.childrenIndexes.get(i));
				}
			}

			dout.writeUTF("BEGIN_DESCRIPTIONS");
			for (int nodeIdx = 0; nodeIdx < tree.descriptions.size(); nodeIdx++) {
				writeBin(tree.descriptions.getTemp(nodeIdx), dout);
			}
			dout.writeUTF("END_BOOFCV_HIERARCHICAL_VOCABULARY_TREE");
			dout.flush();
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static >
	HierarchicalVocabularyTree loadTreeBin( InputStream in, @Nullable HierarchicalVocabularyTree tree ) {
		var builder = new StringBuilder();
		try {
			String line = UtilIO.readLine(in, builder);
			if (!line.equals("BOOFCV_HIERARCHICAL_VOCABULARY_TREE"))
				throw new IOException("Unexpected first line. line.length=" + line.length());

			String pointType = "";
			String distanceClass = "";
			int dof = 0;
			int numDescriptions = 0;
			int branchFactor = 0;
			int maximumLevel = 0;
			int nodesSize = 0;

			while (true) {
				line = UtilIO.readLine(in, builder);
				if (line.equals("BEGIN_GRAPH"))
					break;
				if (line.startsWith("#"))
					continue;
				String[] words = line.split("\\s");
				if (words[0].equals("branch_factor")) {
					branchFactor = Integer.parseInt(words[1]);
				} else if (words[0].equals("maximum_level")) {
					maximumLevel = Integer.parseInt(words[1]);
				} else if (words[0].equals("nodes.size")) {
					nodesSize = Integer.parseInt(words[1]);
				} else if (words[0].equals("descriptions.size")) {
					numDescriptions = Integer.parseInt(words[1]);
				} else if (words[0].equals("point_type")) {
					pointType = words[1];
				} else if (words[0].equals("point_dof")) {
					dof = Integer.parseInt(words[1]);
				} else if (words[0].equals("distance.name")) {
					distanceClass = words[1];
				}
			}

			TD tuple;
			{
				PointDistance distanceFunction;
				PackedArray descriptions;

				switch (pointType) {
					case "TupleDesc_F64" -> {
						distanceFunction = (PointDistance)new TuplePointDistanceEuclideanSq.F64();
						descriptions = (PackedArray)new PackedTupleBigArray_F64(dof);
						tuple = (TD)new TupleDesc_F64(dof);
					}
					case "TupleDesc_F32" -> {
						distanceFunction = (PointDistance)new TuplePointDistanceEuclideanSq.F32();
						descriptions = (PackedArray)new PackedTupleBigArray_F32(dof);
						tuple = (TD)new TupleDesc_F32(dof);
					}
					case "TupleDesc_U8" -> {
						distanceFunction = (PointDistance)new TuplePointDistanceEuclideanSq.U8();
						descriptions = (PackedArray)new PackedTupleBigArray_U8(dof);
						tuple = (TD)new TupleDesc_U8(dof);
					}
					case "TupleDesc_S8" -> {
						distanceFunction = (PointDistance)new TuplePointDistanceEuclideanSq.S8();
						descriptions = (PackedArray)new PackedTupleBigArray_U8(dof);
						tuple = (TD)new TupleDesc_S8(dof);
					}
					case "TupleDesc_B" -> {
						distanceFunction = (PointDistance)new TuplePointDistanceHamming();
						descriptions = (PackedArray)new PackedTupleBigArray_B(dof);
						tuple = (TD)new TupleDesc_B(dof);
					}
					default -> throw new IOException("Unknown point type. " + pointType);
				}

				if (tree == null)
					tree = new HierarchicalVocabularyTree<>(distanceFunction, descriptions);
				else {
					tree.distanceFunction = distanceFunction;
				}

				if (!tree.distanceFunction.getClass().getName().equals(distanceClass))
					throw new IOException("Distance functions do not match: Expected=" + distanceClass);
			}

			tree.branchFactor = branchFactor;
			tree.maximumLevel = maximumLevel;
			tree.nodes.resize(nodesSize);

			DataInputStream input = new DataInputStream(in);
			for (int nodeIdx = 0; nodeIdx < tree.nodes.size; nodeIdx++) {
				HierarchicalVocabularyTree.Node n = tree.nodes.get(nodeIdx);
				n.index = input.readInt();
				n.parent = input.readInt();
				n.branch = input.readInt();
				n.descIdx = input.readInt();
				n.userIdx = input.readInt();
				n.weight = input.readDouble();
				n.childrenIndexes.resize(input.readInt());
				for (int i = 0; i < n.childrenIndexes.size; i++) {
					n.childrenIndexes.data[i] = input.readInt();
				}
			}

			readCheckUTF(input, "BEGIN_DESCRIPTIONS");

			for (int i = 0; i < numDescriptions; i++) {
				readBin(tuple, input);
				tree.descriptions.append(tuple);
			}

			readCheckUTF(input, "END_BOOFCV_HIERARCHICAL_VOCABULARY_TREE");

			return tree;
		} catch (IOException e) {
			throw new RuntimeException(e);
		}
	}

	/**
	 * Saves a TupleDesc dictionary in a binary format
	 *
	 * @param dictionary (Input) Tree that's saved
	 * @param out (Output) Stream the tree is written to
	 */
	public static > void saveDictionaryBin( List dictionary,
																	 int dof, Class descType,
																	 OutputStream out ) {
		String header = "BOOFCV_TUPLE_DICTIONARY\n";
		header += "# tuple format: raw array used internally\n";
		header += "format_version 1\n";
		header += "boofcv_version " + BoofVersion.VERSION + "\n";
		header += "git_sha " + BoofVersion.GIT_SHA + "\n";
		header += "point_type " + descType.getSimpleName() + "\n";
		header += "point_dof " + dof + "\n";
		header += "size " + dictionary.size() + "\n";
		header += "BEGIN_DICTIONARY\n";
		try {
			out.write(header.getBytes(StandardCharsets.UTF_8));

			DataOutputStream dout = new DataOutputStream(out);
			for (int wordIdx = 0; wordIdx < dictionary.size(); wordIdx++) {
				writeBin(dictionary.get(wordIdx), dout);
			}
			dout.writeUTF("END_BOOFCV_TUPLE_DICTIONARY");
			dout.flush();
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static >
	List loadDictionaryBin( InputStream in ) {
		var dictionary = new ArrayList();
		var builder = new StringBuilder();
		try {
			String line = UtilIO.readLine(in, builder);
			if (!line.equals("BOOFCV_TUPLE_DICTIONARY"))
				throw new IOException("Unexpected first line. line.length=" + line.length());

			String pointType = "";
			int dof = 0;
			int size = 0;

			while (true) {
				line = UtilIO.readLine(in, builder);
				if (line.equals("BEGIN_DICTIONARY"))
					break;
				if (line.startsWith("#"))
					continue;
				String[] words = line.split("\\s");
				if (words[0].equals("point_type")) {
					pointType = words[1];
				} else if (words[0].equals("point_dof")) {
					dof = Integer.parseInt(words[1]);
				} else if (words[0].equals("size")) {
					size = Integer.parseInt(words[1]);
				}
			}

			TD tuple;
			switch (pointType) {
				case "TupleDesc_F64" -> tuple = (TD)new TupleDesc_F64(dof);
				case "TupleDesc_F32" -> tuple = (TD)new TupleDesc_F32(dof);
				case "TupleDesc_U8" -> tuple = (TD)new TupleDesc_U8(dof);
				case "TupleDesc_S8" -> tuple = (TD)new TupleDesc_S8(dof);
				case "TupleDesc_B" -> tuple = (TD)new TupleDesc_B(dof);
				default -> throw new IOException("Unknown point type. " + pointType);
			}

			DataInputStream input = new DataInputStream(in);

			for (int i = 0; i < size; i++) {
				readBin(tuple, input);
				dictionary.add(tuple.copy());
			}

			readCheckUTF(input, "END_BOOFCV_TUPLE_DICTIONARY");

			return dictionary;
		} catch (IOException e) {
			throw new RuntimeException(e);
		}
	}

	public static void saveNearestNeighborBin( RecognitionNearestNeighborInvertedFile nn, OutputStream out ) {
		FastAccess inverted = nn.getInvertedFiles();
		BigDogArray_I32 imageDB = nn.getImagesDB();

		String header = "BOOFCV_RECOGNITION_NEAREST_NEIGHBOR\n";
		header += "# inverted files: (int=size), array [int=index, float=weights]\n";
		header += "format_version 1\n";
		header += "boofcv_version " + BoofVersion.VERSION + "\n";
		header += "git_sha " + BoofVersion.GIT_SHA + "\n";
		header += "images.size " + imageDB.size + "\n";
		header += "inverted.size " + inverted.size() + "\n";
		header += "BEGIN_INVERTED\n";
		try {
			out.write(header.getBytes(StandardCharsets.UTF_8));

			DataOutputStream dout = new DataOutputStream(out);
			for (int invertedIdx = 0; invertedIdx < inverted.size(); invertedIdx++) {
				InvertedFile inv = inverted.get(invertedIdx);
				dout.writeInt(inv.size);
				for (int imageIdx = 0; imageIdx < inv.size; imageIdx++) {
					dout.writeInt(inv.get(imageIdx));
					dout.writeFloat(inv.weights.get(imageIdx));
				}
			}
			dout.writeUTF("BEGIN_IMAGES");
			imageDB.forEach(0, imageDB.size, ( index ) -> {
				try {
					dout.writeInt(index);
				} catch (IOException e) {
					throw new UncheckedIOException(e);
				}
			});
			dout.writeUTF("END_BOOFCV_RECOGNITION_NEAREST_NEIGHBOR");
			dout.flush();
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static void loadNearestNeighborBin( InputStream in, RecognitionNearestNeighborInvertedFile nn ) {
		DogArray inverted = nn.getInvertedFiles();
		BigDogArray_I32 imageDB = nn.getImagesDB();

		inverted.reset();
		imageDB.reset();

		var builder = new StringBuilder();
		try {
			String line = UtilIO.readLine(in, builder);
			if (!line.equals("BOOFCV_RECOGNITION_NEAREST_NEIGHBOR"))
				throw new IOException("Unexpected first line. line.length=" + line.length());

			int invertedCount = 0;
			int imageCount = 0;

			while (true) {
				line = UtilIO.readLine(in, builder);
				if (line.startsWith("BEGIN_INVERTED"))
					break;
				if (line.startsWith("#"))
					continue;
				String[] words = line.split("\\s");
				if (words[0].equals("inverted.size")) {
					invertedCount = Integer.parseInt(words[1]);
				} else if (words[0].equals("images.size")) {
					imageCount = Integer.parseInt(words[1]);
				}
			}

			DataInputStream input = new DataInputStream(in);
			for (int invertedIdx = 0; invertedIdx < invertedCount; invertedIdx++) {
				InvertedFile inv = inverted.grow();
				int fileCount = input.readInt();

				for (int imageIdx = 0; imageIdx < fileCount; imageIdx++) {
					int index = input.readInt();
					float weight = input.readFloat();
					inv.addImage(index, weight);
				}
			}

			readCheckUTF(input, "BEGIN_IMAGES");
			for (int imageIdx = 0; imageIdx < imageCount; imageIdx++) {
				imageDB.add(input.readInt());
			}

			readCheckUTF(input, "END_BOOFCV_RECOGNITION_NEAREST_NEIGHBOR");
		} catch (IOException e) {
			throw new RuntimeException(e);
		}
	}

	public static > void writeBin( TD tuple, DataOutputStream dout ) throws IOException {
		if (tuple instanceof TupleDesc_F64) {
			var desc = (TupleDesc_F64)tuple;
			for (int i = 0; i < desc.size(); i++) {
				dout.writeDouble(desc.data[i]);
			}
		} else if (tuple instanceof TupleDesc_F32) {
			var desc = (TupleDesc_F32)tuple;
			for (int i = 0; i < desc.size(); i++) {
				dout.writeFloat(desc.data[i]);
			}
		} else if (tuple instanceof TupleDesc_I8) {
			var desc = (TupleDesc_I8)tuple;
			dout.write(desc.data, 0, desc.size());
		} else if (tuple instanceof TupleDesc_B) {
			var desc = (TupleDesc_B)tuple;
			for (int i = 0; i < desc.data.length; i++) {
				dout.writeInt(desc.data[i]);
			}
		} else {
			throw new IllegalArgumentException("Unknown type " + tuple.getClass().getSimpleName());
		}
	}

	public static > void readBin( TD tuple, DataInputStream in ) throws IOException {
		if (tuple instanceof TupleDesc_F64) {
			var desc = (TupleDesc_F64)tuple;
			for (int i = 0; i < desc.size(); i++) {
				desc.data[i] = in.readDouble();
			}
		} else if (tuple instanceof TupleDesc_F32) {
			var desc = (TupleDesc_F32)tuple;
			for (int i = 0; i < desc.size(); i++) {
				desc.data[i] = in.readFloat();
			}
		} else if (tuple instanceof TupleDesc_I8) {
			var desc = (TupleDesc_I8)tuple;
			BoofMiscOps.checkEq(desc.data.length, in.read(desc.data, 0, desc.data.length));
		} else if (tuple instanceof TupleDesc_B) {
			var desc = (TupleDesc_B)tuple;
			for (int i = 0; i < desc.data.length; i++) {
				desc.data[i] = in.readInt();
			}
		} else {
			throw new IllegalArgumentException("Unknown type " + tuple.getClass().getSimpleName());
		}
	}

	public static > void saveTreeBin( RecognitionVocabularyTreeNister2006 db, File file ) {
		try {
			var out = new BufferedOutputStream(new FileOutputStream(file), 1024*1024);
			saveBin(db, out);
			out.close();
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	public static >
	void loadTreeBin( File file, RecognitionVocabularyTreeNister2006 db ) {
		try {
			var in = new BufferedInputStream(new FileInputStream(file), 1024*1024);
			loadBin(in, db);
			in.close();
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	/**
	 * Saves {@link RecognitionVocabularyTreeNister2006} to a binary format.
	 *
	 * @param db (Input) Structure to be encoded
	 * @param out Stream it's written to
	 */
	public static > void saveBin( RecognitionVocabularyTreeNister2006 db, OutputStream out ) {
		HierarchicalVocabularyTree tree = db.getTree();
		Objects.requireNonNull(tree, "Tree must be specified before it can be saved");

		String header = "BOOFCV_RECOGNITION_NISTER_2006\n";
		header += "# Image DB: id=int,descTermFreq.size=int,array[key=int,value=float]\n";
		header += "# Leaf Info: images.size=int,images.data=array[int]\n";
		header += "format_version 1\n";
		header += "boofcv_version " + BoofVersion.VERSION + "\n";
		header += "git_sha " + BoofVersion.GIT_SHA + "\n";
		header += "images_db.size " + db.getImagesDB().size + "\n";
		header += "BEGIN_TREE\n";

		try {
			out.write(header.getBytes(StandardCharsets.UTF_8));

			// Save the tree next since we need that structure to decode everything that comes after it
			saveTreeBin(tree, out);

			var dout = new DataOutputStream(out);
			dout.writeUTF("BEGIN_IMAGE_DB");
			BigDogArray_I32 imageDB = db.getImagesDB();
			for (int dbIdx = 0; dbIdx < imageDB.size; dbIdx++) {
				dout.writeInt(imageDB.get(dbIdx));
			}

			dout.writeUTF("BEGIN_INVERTED_FILES");
			BoofMiscOps.checkEq(db.invertedFiles.size(), tree.nodes.size);
			for (int nodeIdx = 0; nodeIdx < db.invertedFiles.size(); nodeIdx++) {
				InvertedFile node = db.invertedFiles.get(nodeIdx);
				BoofMiscOps.checkEq(node.size, node.weights.size);

				dout.writeInt(node.size());
				for (int i = 0; i < node.size; i++) {
					dout.writeInt(node.get(i));
				}
				for (int i = 0; i < node.weights.size; i++) {
					dout.writeFloat(node.weights.get(i));
				}
			}

			dout.writeUTF("END BOOFCV_RECOGNITION_NISTER_2006");
			dout.flush();
		} catch (IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	/**
	 * Loads {@link RecognitionVocabularyTreeNister2006} from a binary format.
	 *
	 * @param in Input stream
	 * @param db (Ouput) Decoded structure
	 */
	public static > void loadBin( InputStream in, RecognitionVocabularyTreeNister2006 db ) {
		var builder = new StringBuilder();

		try {
			String line = UtilIO.readLine(in, builder);
			if (!line.equals("BOOFCV_RECOGNITION_NISTER_2006"))
				throw new IOException("Unexpected first line. line.length=" + line.length());

			BigDogArray_I32 imagesDB = db.getImagesDB();
			while (true) {
				line = UtilIO.readLine(in, builder);
				if (line.startsWith("BEGIN_TREE"))
					break;
				if (line.startsWith("#"))
					continue;
				String[] words = line.split("\\s");
				if (words[0].equals("images_db.size")) {
					imagesDB.resize(Integer.parseInt(words[1]));
				}
			}

			db.tree = loadTreeBin(in, null);

			var input = new DataInputStream(in);
			readCheckUTF(input, "BEGIN_IMAGE_DB");
			for (int i = 0; i < imagesDB.size; i++) {
				imagesDB.set(i, input.readInt());
			}

			readCheckUTF(input, "BEGIN_INVERTED_FILES");
			db.invertedFiles.reset();
			db.invertedFiles.resize(db.tree.nodes.size());
			for (int nodeIdx = 0; nodeIdx < db.invertedFiles.size(); nodeIdx++) {
				final InvertedFile node = db.invertedFiles.get(nodeIdx);
				final int N = input.readInt();
				node.resize(N);
				node.weights.resize(N);
				for (int i = 0; i < N; i++) {
					node.set(i, input.readInt());
				}
				for (int i = 0; i < N; i++) {
					node.weights.set(i, input.readFloat());
				}
			}

			readCheckUTF(input, "END BOOFCV_RECOGNITION_NISTER_2006");
		} catch (IOException e) {
			throw new RuntimeException(e);
		}
	}

	private static void readCheckUTF( DataInputStream input, String expected ) throws IOException {
		String line = input.readUTF();
		if (!line.equals(expected))
			throw new IOException("Expected '" + expected + "' not '" + line + "'");
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy