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.wms.location;
import jakarta.persistence.Column;
import jakarta.persistence.Embeddable;
import jakarta.validation.constraints.Max;
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)
    @Max(value = KEY_LENGTH)
    private String area;
    /** Expresses the aisle the {@link Location} belongs to. */
    @Column(name = "C_AISLE", nullable = false, length = KEY_LENGTH)
    @Max(value = KEY_LENGTH)
    private String aisle;
    /** Expresses the x-dimension the {@link Location} belongs to. */
    @Column(name = "C_X", nullable = false, length = KEY_LENGTH)
    @Max(value = KEY_LENGTH)
    private String x;
    /** Expresses the y-dimension the {@link Location} belongs to. */
    @Column(name = "C_Y", nullable = false, length = KEY_LENGTH)
    @Max(value = KEY_LENGTH)
    private String y;
    /** Expresses the z-dimension the {@link Location} belongs to. */
    @Column(name = "C_Z", nullable = false, length = KEY_LENGTH)
    @Max(value = KEY_LENGTH)
    private String z;
    /*~ ----------------------------- constructors ------------------- */
    /** Dear JPA ... */
    protected LocationPK() {
        super();
    }
    /**
     * 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 LocationPK(String area, String aisle, String x, String y, String z) {
        this.area = area;
        this.aisle = aisle;
        this.x = x;
        this.y = y;
        this.z = 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}
     */
    public 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];
    }
    private LocationPK(Builder builder) {
        area = builder.area;
        aisle = builder.aisle;
        x = builder.x;
        y = builder.y;
        z = builder.z;
    }
    public static LocationPK fromString(String s) {
        Assert.hasText(s, "Location String must be provided");
        return new LocationPK(s.split("/"));
    }
    public static boolean isValid(String locationPk) {
        return locationPk != null && locationPk.split("/").length == NUMBER_OF_KEYS;
    }
    /**
     * Create a new builder instance to create LocationPK from.
     *
     * @return The builder
     */
    public static Builder newBuilder() {
        return new Builder();
    }
    /*~ ----------------------------- 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)) {
            return false;
        }
        LocationPK other = (LocationPK) o;
        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;
    }
    /**
     * {@code LocationPK} builder static inner class.
     */
    public static final class Builder {
        private String area;
        private String aisle;
        private String x;
        private String y;
        private String z;
        /**
         * Sets the {@code area} and returns a reference to this Builder so that the methods can be chained together.
         *
         * @param val the {@code area} to set
         * @return a reference to this Builder
         */
        public Builder area(String val) {
            area = val;
            return this;
        }
        /**
         * Sets the {@code aisle} and returns a reference to this Builder so that the methods can be chained together.
         *
         * @param val the {@code aisle} to set
         * @return a reference to this Builder
         */
        public Builder aisle(String val) {
            aisle = val;
            return this;
        }
        /**
         * Sets the {@code x} and returns a reference to this Builder so that the methods can be chained together.
         *
         * @param val the {@code x} to set
         * @return a reference to this Builder
         */
        public Builder x(String val) {
            x = val;
            return this;
        }
        /**
         * Sets the {@code y} and returns a reference to this Builder so that the methods can be chained together.
         *
         * @param val the {@code y} to set
         * @return a reference to this Builder
         */
        public Builder y(String val) {
            y = val;
            return this;
        }
        /**
         * Sets the {@code z} and returns a reference to this Builder so that the methods can be chained together.
         *
         * @param val the {@code z} to set
         * @return a reference to this Builder
         */
        public Builder z(String val) {
            z = val;
            return this;
        }
        /**
         * Returns a {@code LocationPK} built from the parameters previously set.
         *
         * @return a {@code LocationPK} built with parameters of this {@code LocationPK.Builder}
         */
        public LocationPK build() {
            return new LocationPK(this);
        }
    }
}