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

io.activej.ot.OTCommit Maven / Gradle / Ivy

Go to download

Implementation of operational transformation technology. Allows building collaborative software systems.

There is a newer version: 6.0-rc2
Show newest version
/*
 * Copyright (C) 2020 ActiveJ 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.activej.ot;

import io.activej.ot.OTCommitFactory.DiffsWithLevel;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.time.Instant;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Function;
import java.util.function.ToLongFunction;

import static io.activej.common.Checks.checkState;
import static io.activej.common.Utils.entriesToMap;
import static io.activej.common.Utils.keysToMap;
import static java.util.Collections.emptyMap;
import static java.util.Collections.singletonMap;

public final class OTCommit {
	public static final int INITIAL_EPOCH = 0;
	private final int epoch;
	private final K id;
	private final Map> parentsWithLevels;
	private final Map> parents;
	private final long level;

	private long timestamp;
	private byte @Nullable [] serializedData;

	private OTCommit(int epoch, K id, Map> parents) {
		this.epoch = epoch;
		this.id = id;
		this.parentsWithLevels = parents;
		this.parents = entriesToMap(parentsWithLevels.entrySet().stream(), DiffsWithLevel::getDiffs);
		this.level = parents.values().stream().mapToLong(DiffsWithLevel::getLevel).max().orElse(0L) + 1L;
	}

	public static  OTCommit ofRoot(@NotNull K id) {
		return new OTCommit<>(INITIAL_EPOCH, id, emptyMap());
	}

	public static  OTCommit of(int epoch, @NotNull K id, @NotNull Map> parents) {
		return new OTCommit<>(epoch, id, parents);
	}

	public static  OTCommit of(int epoch, K id, Set parents, Function> diffs, ToLongFunction levels) {
		return of(epoch, id, keysToMap(parents.stream(), parent -> new DiffsWithLevel<>(levels.applyAsLong(parent), diffs.apply(parent))));
	}

	public static  OTCommit ofCommit(int epoch, @NotNull K id, @NotNull K parent, @NotNull DiffsWithLevel diffs) {
		return new OTCommit<>(epoch, id, singletonMap(parent, diffs));
	}

	public static  OTCommit ofCommit(int epoch, K id, K parent, List diffs, long level) {
		return ofCommit(epoch, id, parent, new DiffsWithLevel<>(level, diffs));
	}

	public OTCommit withTimestamp(long timestamp) {
		this.timestamp = timestamp;
		return this;
	}

	public OTCommit withSerializedData(byte[] serializedData) {
		this.serializedData = serializedData;
		return this;
	}

	public boolean isRoot() {
		return parents.isEmpty();
	}

	public boolean isMerge() {
		return parents.size() > 1;
	}

	public boolean isCommit() {
		return parents.size() == 1;
	}

	public long getLevel() {
		return level;
	}

	public K getId() {
		return id;
	}

	public int getEpoch() {
		return epoch;
	}

	public Map> getParents() {
		return parents;
	}

	public Map> getParentsWithLevels() {
		return parentsWithLevels;
	}

	public Set getParentIds() {
		return parents.keySet();
	}

	public long getTimestamp() {
		return timestamp;
	}

	public Instant getInstant() {
		checkState(timestamp != 0L, "Timestamp has not been set");
		return Instant.ofEpochMilli(timestamp);
	}

	public byte @Nullable [] getSerializedData() {
		return serializedData;
	}

	public void setSerializedData(byte @Nullable [] serializedData) {
		this.serializedData = serializedData;
	}

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

	@Override
	public int hashCode() {
		return id.hashCode();
	}

	@Override
	public String toString() {
		return "{id=" + id + ", parents=" + getParentIds() + "}";
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy