LocationPK.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;

import jakarta.persistence.Column;
import jakarta.persistence.Embeddable;
import jakarta.validation.constraints.Size;
import org.springframework.util.Assert;

import java.io.Serializable;

/**
 * A LocationPK, is a value type and is used as an unique natural key for {@link Location} entities.
 *
 * @author Heiko Scherrer
 * @see Location
 */
@Embeddable
public class LocationPK implements Serializable {

    public static final short KEY_LENGTH = 4;
    public static final short NUMBER_OF_KEYS = 5;
    /**
     * Returns the complete length of all keys. Currently, all keys have the same length, therefore it is the 5 times the length of a single
     * key (KEY_LENGTH). But since this can change the actual length is encapsulated within this method.
     */
    public static final short PK_LENGTH = NUMBER_OF_KEYS * KEY_LENGTH;

    /** Expresses the area the {@link Location} belongs to. */
    @Column(name = "C_AREA", nullable = false, length = KEY_LENGTH)
    @Size(max = KEY_LENGTH)
    private String area;

    /** Expresses the aisle the {@link Location} belongs to. */
    @Column(name = "C_AISLE", nullable = false, length = KEY_LENGTH)
    @Size(max = KEY_LENGTH)
    private String aisle;

    /** Expresses the x-dimension the {@link Location} belongs to. */
    @Column(name = "C_X", nullable = false, length = KEY_LENGTH)
    @Size(max = KEY_LENGTH)
    private String x;

    /** Expresses the y-dimension the {@link Location} belongs to. */
    @Column(name = "C_Y", nullable = false, length = KEY_LENGTH)
    @Size(max = KEY_LENGTH)
    private String y;

    /** Expresses the z-dimension the {@link Location} belongs to. */
    @Column(name = "C_Z", nullable = false, length = KEY_LENGTH)
    @Size(max = KEY_LENGTH)
    private String z;

    /*~ ----------------------------- constructors ------------------- */

    /** Dear JPA ... */
    protected LocationPK() { }

    /**
     * Create a new LocationPK with all required fields.
     *
     * @param area Area where the {@link Location} belongs to
     * @param aisle Aisle where the {@link Location} belongs to
     * @param x Dimension x where the {@link Location} belongs to
     * @param y Dimension y where the {@link Location} belongs to
     * @param z Dimension z where the {@link Location} belongs to
     */
    public static LocationPK of(String area, String aisle, String x, String y, String z) {
        return new LocationPK(area, aisle, x, y, z);
    }

    /**
     * Weak constructor to create a new LocationPK with a couple of keys only.
     *
     * @param keys The array of keys, currently expected to be 5
     * @throws IllegalArgumentException if the number of keys does not match {@link LocationPK#NUMBER_OF_KEYS}
     */
    private LocationPK(String... keys) {
        if (keys == null || keys.length != NUMBER_OF_KEYS) {
            throw new IllegalArgumentException(
                    "Number of key fields to create a LocationPK does not match the defined number of keys. Expected: " + NUMBER_OF_KEYS);
        }
        this.area = keys[0];
        this.aisle = keys[1];
        this.x = keys[2];
        this.y = keys[3];
        this.z = keys[4];
    }

    /**
     * Create a {@link LocationPK} from the given String.
     *
     * @param s The String, not {@literal null}
     * @return An instance
     */
    public static LocationPK fromString(String s) {
        Assert.hasText(s, "Location String must not be null");
        return new LocationPK(s.split("/"));
    }

    /**
     * Checks whether the given {@code locationPK} String is in valid format.
     *
     * @param locationPk The String to verify
     * @return {@literal true} if valid
     */
    public static boolean isValid(String locationPk) {
        return locationPk != null && locationPk.split("/").length == NUMBER_OF_KEYS;
    }
    /*~ ----------------------------- methods ------------------- */

    /**
     * Get the area region.
     *
     * @return The area
     */
    public String getArea() {
        return this.area;
    }

    /**
     * Get the aisle region.
     *
     * @return The aisle
     */
    public String getAisle() {
        return this.aisle;
    }

    /**
     * Get the x-dimension.
     *
     * @return The x-dimension
     */
    public String getX() {
        return this.x;
    }

    /**
     * Get the y-dimension.
     *
     * @return The y-dimension
     */
    public String getY() {
        return this.y;
    }

    /**
     * Get the z-dimension.
     *
     * @return The z-dimension
     */
    public String getZ() {
        return this.z;
    }

    /**
     * {@inheritDoc}
     *
     * @see Object#equals(Object)
     */
    @Override
    public boolean equals(Object o) {
        if (o == this) {
            return true;
        }
        if (!(o instanceof LocationPK other)) {
            return false;
        }
        return this.y.equals(other.y) && this.x.equals(other.x) && this.area.equals(other.area)
                && this.z.equals(other.z) && this.aisle.equals(other.aisle);
    }

    /**
     * {@inheritDoc}
     *
     * @see Object#hashCode()
     */
    @Override
    public int hashCode() {
        return this.y.hashCode() ^ this.x.hashCode() ^ this.area.hashCode() ^ this.z.hashCode() ^ this.aisle.hashCode();
    }

    /**
     * Return a String like AREA/AISLE/X/Y/Z.
     *
     * @return String
     * @see Object#toString()
     */
    @Override
    public String toString() {
        return this.area + "/" + this.aisle + "/" + this.x + "/" + this.y + "/" + this.z;
    }
}