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

com.credibledoc.iso8583packer.validator.ValidatorService Maven / Gradle / Ivy

package com.credibledoc.iso8583packer.validator;

import com.credibledoc.iso8583packer.dump.Visualizer;
import com.credibledoc.iso8583packer.exception.PackerRuntimeException;
import com.credibledoc.iso8583packer.hex.HexService;
import com.credibledoc.iso8583packer.length.LengthPacker;
import com.credibledoc.iso8583packer.message.MsgField;
import com.credibledoc.iso8583packer.message.MsgFieldType;
import com.credibledoc.iso8583packer.navigator.Navigator;
import com.credibledoc.iso8583packer.tag.TagPacker;

import java.util.Arrays;
import java.util.List;
import java.util.Objects;

/**
 * The service is used for validation the {@link com.credibledoc.iso8583packer.message.MsgField} structure.
 * 
 * @author Kyrylo Semenko
 */
public class ValidatorService implements Validator {
    
    protected Navigator navigator;
    protected Visualizer visualizer;

    @Override
    public void validateStructure(MsgField current) {
        validateStructureRecursively(current);
    }

    /**
     * Check restrictions of the Field structure and all its children recursively.
     *
     * @param msgField the {@link MsgField} to be checked.
     */
    protected void validateStructureRecursively(MsgField msgField) {
        if (msgField == null) {
            throw new PackerRuntimeException("Field is null");
        }

        if (msgField.getType() == null) {
            throw new PackerRuntimeException("Please set the " + MsgFieldType.class.getSimpleName() +
                " value to the field with path: '" + navigator.getPathRecursively(msgField) + "'");
        }
        
        if (msgField.getTag() == null && msgField.getName() == null && msgField.getType() != MsgFieldType.VAL) {
            throw new PackerRuntimeException("At least one 'tag' or 'name' should be set to the field " +
                    "but the both properties are 'null'. Field path: '" + navigator.getPathRecursively(msgField) + "'");
        }

        if (msgField.getBodyPacker() == null && msgField.getChildren() == null &&
                msgField.getType() != MsgFieldType.BIT_SET && msgField.getType() != MsgFieldType.MSG) {
            throw new PackerRuntimeException("Please call the 'defineBodyPacker(...)' method " +
                    "because the 'BodyPacker' value is mandatory. " +
                    "Field path: '" + navigator.getPathRecursively(msgField) + "'.");
        }

        if (MsgFieldType.isTaggedType(msgField)) {
            validateTagAndTagPackerExists(msgField);
        }
        
        if (msgField.getType() == MsgFieldType.TAG_LEN_VAL || msgField.getType() == MsgFieldType.LEN_TAG_VAL) {
            validateHeaderOrParentLengthPackerExists(msgField);
            validateHasNoBitSetAndBitMapPacker(msgField);
        }
        
        if (msgField.getType() == MsgFieldType.TAG_VAL) {
            validateHasNoBitSetAndBitMapPacker(msgField);
        }
        
        if (msgField.getType() == MsgFieldType.LEN_VAL) {
            validateHeaderOrParentLengthPackerExists(msgField);
            validateHasNoBitSetAndBitMapPacker(msgField);
        }
        
        if (msgField.getType() == MsgFieldType.VAL) {
            validateHasNoBitSetAndBitMapPacker(msgField);
        }
        
        if (msgField.getType() == MsgFieldType.BIT_SET) {
            validateBitSetAndBitMapPackerExists(msgField);
            validateChildrenExists(msgField);
        }

        validateMsgType(msgField);

        validateFixedLenSubfields(msgField);
        
        validateFixedLengthType(msgField);

        validateChildren(msgField);

        if (msgField.getRoot() == null) {
            throw new PackerRuntimeException("The reference to a root node is 'null'. " +
                "Field path: '" + navigator.getPathRecursively(msgField) + "'.");
        }
    }

    private void validateMsgType(MsgField msgField) {
        if (msgField.getType() == MsgFieldType.MSG) {
            validateHasNoBitSetAndBitMapPacker(msgField);
            validateHasNoLenDefined(msgField);
        }
    }

    private void validateChildren(MsgField msgField) {
        List msgFields = msgField.getChildren();

        if (msgFields != null) {
            validateHasNoBodyPacker(msgField);
        }

        if (msgFields != null) {
            for (int i = 0; i < msgFields.size(); i++) {
                MsgField nextMsgField = msgFields.get(i);
                validateStructureRecursively(nextMsgField);
                if (i < msgFields.size() - 1) {
                    // all except the last child
                    validateLenExists(nextMsgField);
                }
            }
        }
    }

    void validateFixedLengthType(MsgField msgField) {
        MsgFieldType type = msgField.getType();
        if (msgField.getLen() != null && MsgFieldType.isLengthType(msgField)) {
            throw new PackerRuntimeException("The current MsgField with type " + type + " and path '" +
                navigator.getPathRecursively(msgField) +
                "' cannot have the 'fieldLen'" +
                " value because the type belongs to the following types: " +
                Arrays.toString(MsgFieldType.getLengthTypes().toArray()));
        }
    }

    private void validateHasNoBodyPacker(MsgField msgField) {
        if (msgField.getBodyPacker() != null) {
            throw new PackerRuntimeException("The MsgField with path '" + navigator.getPathRecursively(msgField) +
                "' has defined bodyPacker '" + msgField.getBodyPacker().getClass().getSimpleName() +
                "', but it is redundant, because it has children.");
        }
    }

    protected void validateFixedLenSubfields(MsgField msgField) {
        boolean isLenType =
            msgField.getLen() != null &&
            msgField.getChildren() != null &&
            allChildrenAreLenType(msgField.getChildren());
        
        if (isLenType) {
            int childrenLen = calculateChildrenLen(msgField.getChildren());
            if (msgField.getLen() != childrenLen) {
                throw new PackerRuntimeException("The length '" + msgField.getLen() + "' of the field '" +
                    navigator.getPathRecursively(msgField) +
                    "' and the sum of the lengths of its children '" + childrenLen + "' not equal");
            }
        }
    }

    protected int calculateChildrenLen(List children) {
        int result = 0;
        for (MsgField msgField : children) {
            result += msgField.getLen();
        }
        return result;
    }

    protected boolean allChildrenAreLenType(List children) {
        for (MsgField msgField : children) {
            if (msgField.getLen() == null) {
                return false;
            }
        }
        return true;
    }

    protected void validateHasNoLenDefined(MsgField msgField) {
        if (msgField.getLen() != null) {
            throw new PackerRuntimeException("The 'len' property is not allowed for the MsgField with path '" +
                navigator.getPathRecursively(msgField) +
                "' with the '" + MsgFieldType.class.getSimpleName() + "." + msgField.getType() + "' type " +
                "because the 'len' property is not used in such field type.");
        }
    }

    protected void validateHasNoBitSetAndBitMapPacker(MsgField msgField) {
        if (msgField.getBitMapPacker() != null) {
            throw new PackerRuntimeException("BitMapPacker is not allowed for MsgField '" +
                navigator.getPathRecursively(msgField) +
                "' with MsgType '" + msgField.getType() + "' because that doesn't make sense.");
        }
    }

    protected void validateTagAndTagPackerExists(MsgField msgField) {
        boolean parentTagPackerExists = msgField.getParent() != null && msgField.getParent().getChildrenTagPacker() != null;
        if (parentTagPackerExists && msgField.getTagPacker() != null) {
            String parentPath = navigator.getPathRecursively(msgField.getParent());
            throw new PackerRuntimeException("Only one TagPacker definition is allowed " +
                "for the MsgField with path '" + navigator.getPathRecursively(msgField) +
                "', but found its parent childrenTagPacker with path '" + parentPath +
                "'. Please chose only one TagPacker.");
        }
        
        boolean tagPackerExists = parentTagPackerExists || msgField.getTagPacker() != null;
        Object tag = msgField.getTag();
        boolean tagExists = tag != null;

        if (!tagPackerExists) {
            String parentPath = navigator.getPathRecursively(msgField.getParent());
            String parentString = parentPath == null ? "" : " or its parent with path '" + parentPath + "'";
            throw new PackerRuntimeException("Please define the '" + TagPacker.class.getSimpleName() +
                "' value to the MsgField with path '" + navigator.getPathRecursively(msgField) + "'" +
                parentString + ", " +
                "because it is mandatory for MsgFieldType '" + msgField.getType() + "'. Please call the " +
                "fieldBuilder.defineChildrenTagPacker(..) method for its parent " +
                "or fieldBuilder.defineHeaderTagPacker(..) method for the MsgField itself.");
        }

        if (!tagExists) {
            throw new PackerRuntimeException("Please define the tag value to the field '" +
                navigator.getPathRecursively(msgField) +
                    "', it is mandatory for MsgFieldType '" + msgField.getType() + "'.");
        }

        TagPacker tagPacker = msgField.getTagPacker();
        if (tagPacker == null) {
            tagPacker = msgField.getParent().getChildrenTagPacker();
        }
        byte[] packedTag = tagPacker.pack(tag);
        Object unpackedTag = tagPacker.unpack(packedTag, 0);
        if (!Objects.equals(tag, unpackedTag)) {
            throw new PackerRuntimeException("MsgField with path '" + navigator.getPathRecursively(msgField) +
                "' contains tag '" + tag + "', " +
                "the tag is packed as bytes '" + HexService.bytesToHex(packedTag) + "'. " +
                "But unpacked value '" + unpackedTag + "' of the tag " +
                "not equals with the original value '" + tag + "'." +
                " The tag is packed and unpacked with the '" + tagPacker.getClass().getSimpleName() + "' TagPacker." +
                " Please define other TagPacker.");
        }
    }

    protected void validateBitSetAndBitMapPackerExists(MsgField msgField) {
        if (msgField.getBitMapPacker() == null) {
            throw new PackerRuntimeException("The bitMapPacker value is mandatory for '" + msgField.getType() +
                    "' field type. Please call the defineHeaderBitmapPacker(...) method. Field path: " +
                navigator.getPathRecursively(msgField) + ".");
        }
    }

    protected void validateHeaderOrParentLengthPackerExists(MsgField msgField) {
        boolean parentHasLengthPacker = msgField.getParent() != null && msgField.getParent().getChildrenLengthPacker() != null;
        boolean fieldHasLengthPacker = msgField.getLengthPacker() != null;
        if (!parentHasLengthPacker && !fieldHasLengthPacker) {
            throw new PackerRuntimeException("The '" + LengthPacker.class.getSimpleName() +
                    "' value is mandatory for the field '" + navigator.getPathRecursively(msgField) +
                    "' or its parent because it has the '" + msgField.getType() +
                    "' type. Please call the defineHeaderLengthPacker() method on the field " +
                    "or defineChildrenLengthPacker() method on the field parent.");
        }
    }

    protected void validateLenExists(MsgField msgField) {
        if (msgField.getLen() == null &&
            (msgField.getType() == MsgFieldType.VAL || msgField.getType() == MsgFieldType.TAG_VAL)) {

            String path = navigator.getPathRecursively(msgField);            
                
            throw new PackerRuntimeException("The field with path '" + path +
                "' has '" + msgField.getType() +
                "' type, so please define its length by calling the defineLen() method or change its type.");
        }
    }

    protected void validateChildrenExists(MsgField msgField) {
        if (msgField.getChildren() == null || msgField.getChildren().isEmpty()) {
            throw new PackerRuntimeException("The field '" + navigator.getPathRecursively(msgField) +
                    "' has no children but has the '" + msgField.getType() +
                    "' type. Please define at least one child with the .createChild(...) method.");
        }
    }

    /**
     * @param navigator see the {@link #navigator} field description.
     */
    @Override
    public void setNavigator(Navigator navigator) {
        this.navigator = navigator;
    }

    /**
     * @param visualizer see the {@link #visualizer} field description.
     */
    @Override
    public void setVisualizer(Visualizer visualizer) {
        this.visualizer = visualizer;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy