/* * jPOS Project [http://jpos.org] * Copyright (C) 2000-2017 jPOS Software SRL * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package org.jpos.iso; import java.io.IOException; import java.io.InputStream; /** * @author joconnor * @version $Revision$ $Date$ */ public class ISOFilledStringFieldPackager extends ISOFieldPackager { private Interpreter interpreter; private Padder padder; private Prefixer prefixer; /** * Constructs a default ISOFilledStringFieldPackager. There is no padding, * no length prefix and a literal interpretation. The set methods must be called to * make this useful. */ public ISOFilledStringFieldPackager() { super(); this.padder = NullPadder.INSTANCE; this.interpreter = LiteralInterpreter.INSTANCE; this.prefixer = NullPrefixer.INSTANCE; } /** * Constructs an ISOFilledStringFieldPackager with a specific Padder, Interpreter and Prefixer. * The length and description should be set with setLength() and setDescription methods. * @param padder The type of padding used. * @param interpreter The interpreter used to encode the field. * @param prefixer The type of length prefixer used to encode this field. */ public ISOFilledStringFieldPackager(Padder padder, Interpreter interpreter, Prefixer prefixer) { super(); this.padder = padder; this.interpreter = interpreter; this.prefixer = prefixer; } /** * Creates an ISOFilledStringFieldPackager. * @param maxLength The maximum length of the field in characters or bytes depending on the datatype. * @param description The description of the field. For human readable output. * @param interpreter The interpreter used to encode the field. * @param padder The type of padding used. * @param prefixer The type of length prefixer used to encode this field. */ public ISOFilledStringFieldPackager(int maxLength, String description, Padder padder, Interpreter interpreter, Prefixer prefixer) { super(maxLength, description); this.padder = padder; this.interpreter = interpreter; this.prefixer = prefixer; } /** * Sets the Padder. * @param padder The padder to use during packing and unpacking. */ public void setPadder(Padder padder) { this.padder = padder; } /** * Sets the Interpreter. * @param interpreter The interpreter to use in packing and unpacking. */ public void setInterpreter(Interpreter interpreter) { this.interpreter = interpreter; } /** * Sets the length prefixer. * @param prefixer The length prefixer to use during packing and unpacking. */ public void setPrefixer(Prefixer prefixer) { this.prefixer = prefixer; } /** * Returns the prefixer's packed length and the interpreter's packed length. */ public int getMaxPackedLength() { return prefixer.getPackedLength() + interpreter.getPackedLength(getLength()); } /** Create a nice readable message for errors */ private String makeExceptionMessage(ISOComponent c, String operation) { Object fieldKey = "unknown"; if (c != null) { try { fieldKey = c.getKey(); } catch (Exception ignore) { } } return this.getClass().getName() + ": Problem " + operation + " field " + fieldKey; } /** * Convert the component into a byte[]. */ public byte[] pack(ISOComponent c) throws ISOException { try { String data = (String)c.getValue(); if (data.length() > getLength()) { throw new ISOException("Field length " + data.length() + " too long. Max: " + getLength()); } String paddedData = padder.pad(data, getLength()); byte[] rawData = new byte[prefixer.getPackedLength() + interpreter.getPackedLength(paddedData.length())]; prefixer.encodeLength(data.length(), rawData); interpreter.interpret(paddedData, rawData, prefixer.getPackedLength()); return rawData; } catch(Exception e) { throw new ISOException(makeExceptionMessage(c, "packing"), e); } } /** * Unpacks the byte array into the component. * @param c The component to unpack into. * @param b The byte array to unpack. * @param offset The index in the byte array to start unpacking from. * @return The number of bytes consumed unpacking the component. */ public int unpack(ISOComponent c, byte[] b, int offset) throws ISOException { try { int len = prefixer.decodeLength(b, offset); if (len == -1) { // The prefixer doesn't know how long the field is, so use // maxLength instead len = getLength(); } else if (getLength() > 0 && len > getLength()) throw new ISOException("Field length " + len + " too long. Max: " + getLength()); int lenLen = prefixer.getPackedLength(); String unpacked = interpreter.uninterpret(b, offset + lenLen, len); c.setValue(unpacked); return lenLen + interpreter.getPackedLength(getLength()); } catch(Exception e) { throw new ISOException(makeExceptionMessage(c, "unpacking"), e); } } /** * Unpack the input stream into the component. * @param c The Component to unpack into. * @param in Input stream where the packed bytes come from. * @exception IOException Thrown if there's a problem reading the input stream. */ public void unpack (ISOComponent c, InputStream in) throws IOException, ISOException { try { int lenLen = prefixer.getPackedLength (); int len; if (lenLen == 0) { len = getLength(); } else { len = prefixer.decodeLength (readBytes (in, lenLen), 0); if (getLength() > 0 && len > 0 && len > getLength()) throw new ISOException("Field length " + len + " too long. Max: " + getLength()); } int packedLen = interpreter.getPackedLength(len); String unpacked = interpreter.uninterpret(readBytes (in, packedLen), 0, len); c.setValue(unpacked); in.skip(interpreter.getPackedLength(getLength()) - packedLen); } catch(ISOException e) { throw new ISOException(makeExceptionMessage(c, "unpacking"), e); } } /** * Checks the length of the data against the maximum, and throws an IllegalArgumentException. * This is designed to be called from field Packager constructors and the setLength() * method. * @param len The length of the data for this field packager. * @param maxLength The maximum length allowed for this type of field packager. * This depends on the prefixer that is used. * @throws IllegalArgumentException If len > maxLength. */ protected void checkLength(int len, int maxLength) throws IllegalArgumentException { if (len > maxLength) { throw new IllegalArgumentException("Length " + len + " too long for " + getClass().getName()); } } }