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

io.datakernel.remotefs.FileMetadata Maven / Gradle / Ivy

Go to download

Package provides tools for building efficient, scalable remote file servers. It utilizes CSP for fast and reliable file transfer.

The newest version!
/*
 * Copyright (C) 2015-2019 SoftIndex LLC.
 *
 * 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 io.datakernel.remotefs;

import io.datakernel.common.parse.ParseException;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.util.*;
import java.util.stream.Stream;

import static io.datakernel.common.Preconditions.checkArgument;

/**
 * This is a POJO for holding name, size, timestamp and revision of some file
 */
public final class FileMetadata {
	public static final Comparator COMPARATOR =
			Comparator.comparingLong(FileMetadata::getRevision)
					.thenComparing((a, b) -> a.isTombstone() == b.isTombstone() ? 0 : a.isTombstone() ? 1 : -1)
					.thenComparing(FileMetadata::getSize)
					.thenComparing(FileMetadata::getTimestamp);

	private final String name;
	private final long size;
	private final long timestamp;
	private final long revision;

	private FileMetadata(String name, long size, long timestamp, long revision) {
		this.name = name;
		this.size = size;
		this.timestamp = timestamp;
		this.revision = revision;
	}

	public static FileMetadata of(@NotNull String name, long size, long timestamp, long revision) {
		checkArgument(size >= 0, "size >= 0");
		return new FileMetadata(name, size, timestamp, revision);
	}

	public static FileMetadata parse(String name, long size, long timestamp, long revision) throws ParseException {
		if (name == null) {
			throw new ParseException(FileMetadata.class, "Name is null");
		}
		if (size < -1) {
			throw new ParseException(FileMetadata.class, "Size is less than zero and not -1");
		}
		return new FileMetadata(name, size, timestamp, revision);
	}

	public static FileMetadata tombstone(String name, long timestamp, long revision) {
		return new FileMetadata(name, -1, timestamp, revision);
	}

	public FileMetadata withName(String name) {
		return new FileMetadata(name, size, timestamp, revision);
	}

	public FileMetadata asTombstone() {
		return new FileMetadata(name, -1, timestamp, revision);
	}

	public boolean isTombstone() {
		return size == -1;
	}

	public String getName() {
		return name;
	}

	public long getSize() {
		return size;
	}

	public long getTimestamp() {
		return timestamp;
	}

	public long getRevision() {
		return revision;
	}

	@Override
	public String toString() {
		return name + "(size=" + size + ", timestamp=" + timestamp + ", revision=" + revision + ')';
	}

	@Override
	public boolean equals(Object o) {
		if (this == o) {
			return true;
		}
		if (o == null || getClass() != o.getClass()) {
			return false;
		}

		FileMetadata that = (FileMetadata) o;

		return size == that.size && timestamp == that.timestamp && revision == that.revision && name.equals(that.name);
	}

	@Override
	public int hashCode() {
		return 29791 * name.hashCode()
				+ 961 * ((int) (size ^ (size >>> 32)))
				+ 31 * ((int) (timestamp ^ (timestamp >>> 32)))
				+ (int) (revision ^ (revision >>> 32));
	}

	@Nullable
	public static FileMetadata getMoreCompleteFile(@Nullable FileMetadata first, @Nullable FileMetadata second) {
		return first == null ? second : second == null ? first : COMPARATOR.compare(first, second) > 0 ? first : second;
	}

	public static List flatten(Stream> streamOfLists) {
		Map map = new HashMap<>();
		streamOfLists
				.flatMap(List::stream)
				.forEach(meta -> map.compute(meta.getName(), ($, existing) -> getMoreCompleteFile(existing, meta)));
		return new ArrayList<>(map.values());
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy