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

org.teamapps.universaldb.index.fileng.FileIndex Maven / Gradle / Ivy

/*-
 * ========================LICENSE_START=================================
 * UniversalDB
 * ---
 * Copyright (C) 2014 - 2021 TeamApps.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.
 * =========================LICENSE_END==================================
 */
package org.teamapps.universaldb.index.fileng;

import org.apache.commons.io.IOUtils;
import org.teamapps.universaldb.context.UserContext;
import org.teamapps.universaldb.index.*;
import org.teamapps.universaldb.index.file.FileFilter;
import org.teamapps.universaldb.index.numeric.LongIndex;
import org.teamapps.universaldb.index.numeric.ShortIndex;
import org.teamapps.universaldb.index.text.CollectionTextSearchIndex;
import org.teamapps.universaldb.index.text.TextIndex;
import org.teamapps.universaldb.util.DataStreamUtil;

import java.io.*;
import java.util.BitSet;
import java.util.List;
import java.util.Map;

public class FileIndex extends AbstractIndex implements FileStore {

	private final TextIndex hashIndex;
	private final TextIndex nameIndex;
	private final LongIndex sizeIndex;
	private FileTextContentIndex textContentIndex;
	private ShortIndex versionIndex;
	private FileVersionDataIndex versionDataIndex;
	private final CollectionTextSearchIndex fullTextIndex;

	private final String filePath;
	private final boolean indexFileContent;
	private final boolean indexFileVersions;
	private final String secret;

	private final File localFileStorePath;
	private final RemoteFileStore remoteFileStore;

	public FileIndex(String name, TableIndex table, ColumnType columnType, boolean indexFileContent, boolean indexFileVersions, String secret) {
		super(name, table, columnType, indexFileContent ? FullTextIndexingOptions.INDEXED : FullTextIndexingOptions.NOT_INDEXED);
		this.indexFileContent = indexFileContent;
		this.indexFileVersions = indexFileVersions;
		this.secret = secret;
		hashIndex = new TextIndex(name + "-file-hash", table, columnType, false);
		nameIndex = new TextIndex(name + "-file-name", table, columnType, false);
		sizeIndex = new LongIndex(name + "-file-size", table, columnType);
		if (indexFileContent) {
			textContentIndex = new FileTextContentIndex(table.getDataPath(), name + "-file-fulltext-content");
		}
		if (indexFileVersions) {
			versionIndex = new ShortIndex(name + "-file-version", table, columnType);
			versionDataIndex = new FileVersionDataIndex(name, table);
		}
		fullTextIndex = new CollectionTextSearchIndex(getFullTextIndexPath(), name);
		filePath = getFQN().replace('.', '/');

		localFileStorePath = new File(table.getDataPath(), name + "-file-store"); //todo separate base path?

		remoteFileStore = null; //todo
	}

	@Override
	public IndexType getType() {
		return IndexType.FILE_NG;
	}

	@Override
	public FileValue getGenericValue(int id) {
		return getValue(id);
	}

	@Override
	public void setGenericValue(int id, FileValue value) {
		setValue(id, value);
	}

	@Override
	public void removeValue(int id) {
		setValue(id, null);
	}

	public void removeAllVersions(int id) {
		if (indexFileVersions) {
			//todo
		} else {
			removeValue(id);
		}
	}

	public FileValue getValue(int id) {
		String hash = hashIndex.getValue(id);
		if (hash == null) {
			return null;
		} else {
			String filePath = getFilePath(hash);
			File localFile = getLocalFile(filePath);
			long size = getSize(id);
			if (localFile.exists() && localFile.length() == size) {
				return new LocalStoreFileValue(id, this, localFile);
			} else if (remoteFileStore != null){
				try {
					InputStream inputStream = remoteFileStore.getInputStream(filePath);
					FileUtil.decryptAndDecompress(inputStream, localFile, hash);
					return new LocalStoreFileValue(id, this, localFile);
				} catch (Exception e) {
					e.printStackTrace();
				}
				return null;
			} else {
				return null;
			}
		}
	}



	public void setValue(int id, FileValue value) {
		if (value == null) {
			String hash = hashIndex.getValue(id);
			if (indexFileVersions && hash != null) {
				String fileName = nameIndex.getValue(id);
				long size = sizeIndex.getValue(id);
				short version = versionIndex.getValue(id);
				versionDataIndex.addVersionEntry(id, version, hash, fileName, size);
			}
			hashIndex.setValue(id, null);
			nameIndex.setValue(id, null);
			sizeIndex.setValue(id, 0);
		} else {
			String hash = value.getHash();
			String filePath = getFilePath(hash);
			File localFile = getLocalFile(filePath);
			if (!localFile.exists()) {
				try {
					IOUtils.copy(value.getInputStream(), new BufferedOutputStream(new FileOutputStream(localFile)));
					if(remoteFileStore != null) {
						File tempFile = File.createTempFile("temp", ".bin");
						FileUtil.encrypt(localFile, tempFile, hash);
						remoteFileStore.setFile(filePath, tempFile);
					}
				} catch (Exception e) {
					throw new RuntimeException("Error: could not write file to local store:" + filePath, e);
				}
			}
			nameIndex.setValue(id, value.getFileName());
			sizeIndex.setValue(id, value.getSize());
			if (indexFileVersions) {

			}
			if (indexFileContent) {

			} else {

			}
			hashIndex.setValue(id, hash);
		}
	}

	private String getFilePath(String hash) {
		String pathHash = FileUtil.createHash(hash, secret);
		return filePath + pathHash.substring(0, 2) + "/" + pathHash + ".bin";
	}

	private File getLocalFile(String path) {
		return new File(localFileStorePath, path);
	}

	private void writeLocalFile(String path, FileValue value) throws IOException {
		File localFile = getLocalFile(path);
		if (localFile.exists()) {
			return;
		} else {
			localFile.getParentFile().mkdirs();
			try (BufferedOutputStream input = new BufferedOutputStream(new FileOutputStream(localFile), 4096 * 16);
				 BufferedInputStream output = new BufferedInputStream(value.getInputStream(), 4096 * 16)) {
				IOUtils.copy(output, input, 4096 * 4);
			}
		}
	}


	@Override
	public String getHash(int id) {
		return hashIndex.getValue(id);
	}

	@Override
	public String getFileName(int id) {
		return nameIndex.getValue(id);
	}

	@Override
	public long getSize(int id) {
		return sizeIndex.getValue(id);
	}

	@Override
	public InputStream getInputStream(int id) {
		String hash = hashIndex.getValue(id);
		String filePath = getFilePath(hash);
		File localFile = getLocalFile(filePath);
		if (localFile.exists()) {
			try {
				return new FileInputStream(localFile);
			} catch (FileNotFoundException e) {
				e.printStackTrace();
			}
		} else if (true) {
			//todo S3 store...
		}
		return null;
	}

	@Override
	public File getAsFile(int id) {
		String hash = hashIndex.getValue(id);
		String filePath = getFilePath(hash);
		File localFile = getLocalFile(filePath);
		if (localFile.exists()) {
			return localFile;
		} else if (true) {
			//todo S3 store...
		}
		return null;
	}

	@Override
	public File getFileVersion(int id, int version) { //todo should return FileValue
		if (indexFileVersions) {
			FileVersionEntry versionData = versionDataIndex.getVersionData(id, version);
			if (versionData != null) {
				String hash = versionData.getHash();
				String filePath = getFilePath(hash);
				File localFile = getLocalFile(filePath);
				if (localFile.exists()) {
					return localFile;
				} else if (true) {
					//todo S3 store...
				}
			}
		}
		return null;
	}

	@Override
	public int getVersion(int id) {
		return indexFileVersions ? versionIndex.getValue(id) : 0;
	}

	@Override
	public void writeTransactionValue(FileValue value, DataOutputStream dataOutputStream) throws IOException {

	}

	@Override
	public FileValue readTransactionValue(DataInputStream dataInputStream) throws IOException {
		return null;
	}

	@Override
	public List sortRecords(List sortEntries, boolean ascending, UserContext userContext) {
		return null;
	}

	@Override
	public void dumpIndex(DataOutputStream dataOutputStream, BitSet records) throws IOException {
		for (int id = records.nextSetBit(0); id >= 0; id = records.nextSetBit(id + 1)) {
			String hash = hashIndex.getValue(id);
			if (hash != null) {
				String name = nameIndex.getValue(id);
				long size = sizeIndex.getValue(id);
				if (indexFileVersions) {
					short value = versionIndex.getValue(id);
					Map versionData = versionDataIndex.getVersions(id);
				}
				if (indexFileContent) {
					textContentIndex.getEntryIterator();
					//todo..
				}
				dataOutputStream.writeInt(id);
				DataStreamUtil.writeStringWithLengthHeader(dataOutputStream,hash);
				DataStreamUtil.writeStringWithLengthHeader(dataOutputStream,name);
				dataOutputStream.writeLong(size);
				//DataStreamUtil.writeByteArrayWithLengthHeader(dataOutputStream,metaData);
				//todo...
			}
		}
	}

	@Override
	public void restoreIndex(DataInputStream dataInputStream) throws IOException {
		try {
			int id = dataInputStream.readInt();
			String hash = DataStreamUtil.readStringWithLengthHeader(dataInputStream);
			String name = DataStreamUtil.readStringWithLengthHeader(dataInputStream);
			long size = dataInputStream.readLong();
			//todo...
		} catch (EOFException ignore) {}
	}

	@Override
	public BitSet filter(BitSet records, FileFilter fileFilter) {
		return null;
	}

	@Override
	public void close() {
		fullTextIndex.commit(true);
		hashIndex.close();
		nameIndex.close();
		sizeIndex.close();
		if (textContentIndex != null) {
			textContentIndex.close();
		}
		if (versionIndex != null) {
			versionIndex.close();
			versionDataIndex.close();
		}
	}

	@Override
	public void drop() {
		fullTextIndex.drop();
		hashIndex.drop();
		nameIndex.drop();
		sizeIndex.drop();
		if (textContentIndex != null) {
			textContentIndex.drop();
		}
		if (versionIndex != null) {
			versionIndex.drop();
			versionDataIndex.drop();
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy