LocationGroupVO.java

/*
 * Copyright 2005-2025 the original author or authors.
 *
 * 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 org.openwms.common.location.api;

import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.JsonProperty;
import jakarta.validation.constraints.NotBlank;
import org.springframework.hateoas.RepresentationModel;

import java.io.Serializable;
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.StringJoiner;
import java.util.stream.Stream;

import static org.openwms.common.location.api.LocationApiConstants.DATETIME_FORMAT_ZULU;

/**
 * A LocationGroupVO represents a {@code LocationGroup}.
 *
 * @author Heiko Scherrer
 */
@JsonInclude(JsonInclude.Include.NON_EMPTY)
@JsonAutoDetect(fieldVisibility = JsonAutoDetect.Visibility.PUBLIC_ONLY)
public class LocationGroupVO extends RepresentationModel<LocationGroupVO> implements TargetVO, Serializable {

    /** The persistent technical key of the {@code LocationGroup}. */
    @JsonProperty("pKey")
    private String pKey;

    /** Unique identifier of the {@code LocationGroup}. */
    @NotBlank
    @JsonProperty("name")
    private String name;

    /** The {@code LocationGroup} might be assigned to an {@code Account}. */
    @JsonProperty("accountId")
    private String accountId;

    /** Description of the {@code LocationGroup}. */
    @JsonProperty("description")
    private String description;

    /** A type can be assigned to a {@code LocationGroup}. */
    @JsonProperty("groupType")
    private String groupType;

    /** Parent {@code LocationGroup}. */
    @JsonProperty("parentName")
    private String parent;

    /** The operation mode is controlled by the subsystem and defines the physical mode a {@code LocationGroup} is currently able to operate in. */
    @NotBlank(groups = ValidationGroups.Create.class)
    @JsonProperty("operationMode")
    private String operationMode;

    /** Infeed state, controlled by the subsystem only. */
    @JsonProperty("groupStateIn")
    private LocationGroupState groupStateIn;

    /** Outfeed state. */
    @JsonProperty("groupStateOut")
    private LocationGroupState groupStateOut;

    /** Child {@code LocationGroup}s. */
    @JsonProperty("childLocationGroups")
    private List<LocationGroupVO> children;

    /** Timestamp when the {@code LocationGroup} has been created. */
    @JsonProperty("createDt")
    @JsonFormat(shape = JsonFormat.Shape.STRING, pattern = DATETIME_FORMAT_ZULU) // required
    private LocalDateTime createDt;

    /*~ ------------------ constructors ----------------------*/
    public LocationGroupVO() {}

    public LocationGroupVO(String name) {
        this.name = name;
    }

    public LocationGroupVO(String name, LocationGroupState groupStateIn, LocationGroupState groupStateOut) {
        this.name = name;
        this.groupStateIn = groupStateIn;
        this.groupStateOut = groupStateOut;
    }

    /*~ ------------------ methods ----------------------*/

    /**
     * Creates a new LocationGroupVO object with the given name and operation mode.
     *
     * @param name The name of the location group.
     * @param operationMode The operation mode of the location group.
     * @return The created LocationGroupVO object.
     */
    public static LocationGroupVO create(String name, String operationMode) {
        var result = new LocationGroupVO(name);
        result.setOperationMode(operationMode);
        return result;
    }

    public Stream<LocationGroupVO> streamLocationGroups() {
        return Stream.concat(
                Stream.of(this),
                children == null ? Stream.empty() : children.stream().flatMap(LocationGroupVO::streamLocationGroups));
    }

    /**
     * Check whether the LocationGroup has a parent.
     *
     * @return {@literal true} if it has a parent, otherwise {@literal false}
     */
    public boolean hasParent() {
        return parent != null && !parent.isEmpty();
    }

    /**
     * Checks whether the LocationGroup is blocked for infeed.
     *
     * @return {@literal true} if blocked, otherwise {@literal false}
     */
    @JsonIgnore
    public boolean isInfeedBlocked() {
        return !isIncomingActive();
    }

    /**
     * Checks whether the LocationGroup is available for infeed.
     *
     * @return {@literal true} if available, otherwise {@literal false}
     */
    @JsonIgnore
    public boolean isIncomingActive() {
        return this.groupStateIn == LocationGroupState.AVAILABLE;
    }

    /**
     * Set the infeed mode.
     *
     * @param incomingActive {@literal true} if available for infeed otherwise {@literal false}
     */
    public void setIncomingActive(boolean incomingActive) {
        this.groupStateIn = incomingActive ? LocationGroupState.AVAILABLE : LocationGroupState.NOT_AVAILABLE;
    }

    public void setGroupStateIn(LocationGroupState groupStateIn) {
        this.groupStateIn = groupStateIn;
    }

    /**
     * Checks whether the LocationGroup is available for outfeed.
     *
     * @return {@literal true} if available, otherwise {@literal false}
     */
    @JsonIgnore
    public boolean isOutgoingActive() {
        return this.groupStateOut == LocationGroupState.AVAILABLE;
    }

    /**
     * Set the outfeed mode.
     *
     * @param outgoingActive {@literal true} if available for outfeed otherwise {@literal false}
     */
    public void setOutgoingActive(boolean outgoingActive) {
        this.groupStateIn = outgoingActive ? LocationGroupState.AVAILABLE : LocationGroupState.NOT_AVAILABLE;
    }

    public void setGroupStateOut(LocationGroupState groupStateOut) {
        this.groupStateOut = groupStateOut;
    }

    /*~ ------------------ accessors ----------------------*/
    public String getOperationMode() {
        return operationMode == null ? "" : operationMode;
    }

    public void setOperationMode(String operationMode) {
        this.operationMode = operationMode;
    }

    public LocationGroupState getGroupStateIn() {
        return groupStateIn;
    }

    public LocationGroupState getGroupStateOut() {
        return groupStateOut;
    }

    public String getpKey() {
        return pKey;
    }

    public void setpKey(String pKey) {
        this.pKey = pKey;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAccountId() {
        return accountId;
    }

    public void setAccountId(String accountId) {
        this.accountId = accountId;
    }

    public String getDescription() {
        return description;
    }

    public void setDescription(String description) {
        this.description = description;
    }

    public String getGroupType() {
        return groupType;
    }

    public void setGroupType(String groupType) {
        this.groupType = groupType;
    }

    public String getParent() {
        return parent;
    }

    public void setParent(String parent) {
        this.parent = parent;
    }

    public List<LocationGroupVO> getChildren() {
        return children;
    }

    public LocationGroupVO addChild(LocationGroupVO child) {
        if (child == null) {
            throw new IllegalArgumentException("Child to add must not be null");
        }
        if (children == null) {
            children = new ArrayList<>();
        }
        children.add(child);
        return this;
    }

    public void setChildren(List<LocationGroupVO> children) {
        this.children = children;
    }

    public LocalDateTime getCreateDt() {
        return createDt;
    }

    public void setCreateDt(LocalDateTime createDt) {
        this.createDt = createDt;
    }

    /*~ ------------------ overrides ----------------------*/
    /**
     * {@inheritDoc}
     */
    @Override
    public String asString() {
        return name;
    }

    /**
     * {@inheritDoc}
     *
     * All fields.
     */
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (!(o instanceof LocationGroupVO that)) return false;
        if (!super.equals(o)) return false;
        return Objects.equals(pKey, that.pKey) && Objects.equals(name, that.name) && Objects.equals(accountId, that.accountId) && Objects.equals(description, that.description) && Objects.equals(groupType, that.groupType) && Objects.equals(parent, that.parent) && Objects.equals(operationMode, that.operationMode) && groupStateIn == that.groupStateIn && groupStateOut == that.groupStateOut && Objects.equals(children, that.children) && Objects.equals(createDt, that.createDt);
    }

    /**
     * {@inheritDoc}
     *
     * All fields.
     */
    @Override
    public int hashCode() {
        return Objects.hash(super.hashCode(), pKey, name, accountId, description, groupType, parent, operationMode, groupStateIn, groupStateOut, children, createDt);
    }

    /**
     * {@inheritDoc}
     *
     * Only the name.
     */
    @Override
    public String toString() {
        return name;
    }

    /**
     * Returns a string representation of the LocationGroupVO object, including all its fields.
     * Fields are concatenated using a delimiter ", ".
     *
     * @return A string representation of the LocationGroupVO object.
     */
    public String allFieldsToString() {
        return new StringJoiner(", ", LocationGroupVO.class.getSimpleName() + "[", "]")
                .add("pKey='" + pKey + "'")
                .add("name='" + name + "'")
                .add("accountId='" + accountId + "'")
                .add("description='" + description + "'")
                .add("groupType='" + groupType + "'")
                .add("parent='" + parent + "'")
                .add("operationMode='" + operationMode + "'")
                .add("groupStateIn=" + groupStateIn)
                .add("groupStateOut=" + groupStateOut)
                .add("children=" + children)
                .add("createDt=" + createDt)
                .toString();
    }
}