/**
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.camel.component.smpp;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class SmppSplitter {
/**
* The length of the UDH in bytes.
* <p/>
* The real length of the header must be 6 bytes, but the first byte that
* contains the length of the header must not be counted.
*/
protected static final int UDHIE_HEADER_LENGTH = 0x05;
/**
* The real length of the UDH header.
* <p/>
* The real length of the UDH header is {@link #UDHIE_HEADER_LENGTH}
* {@code + 1}.
*
* @see #UDHIE_HEADER_LENGTH
*/
protected static final int UDHIE_HEADER_REAL_LENGTH = UDHIE_HEADER_LENGTH + 1;
/**
* The length of the reference number of the SAR fragmet of the UDH header.
* <p/>
* The length can be 1 or 2 bytes and is considered to be 1 byte.
*/
protected static final int UDHIE_SAR_REF_NUM_LENGTH = 1;
/**
* The value that identifier length of the SAR fragment.
* <p/>
* {@code 0x00} value must be used if the legnth of the reference number is
* 1 byte.<br/>
* {@code 0x08} value must be used if the legnth of the reference number is
* 2 bytes.
*/
protected static final byte UDHIE_IDENTIFIER_SAR = 0x00;
/**
* The length of the SAR fragment.
* <p/>
* {@code 0x03} value must be used if the legnth of the reference number is
* 1 byte.<br/>
* {@code 0x04} value must be used if the legnth of the reference number is
* 2 bytes.
*/
protected static final byte UDHIE_SAR_LENGTH = 0x03;
/**
* The maximum length of the message in bytes.
*/
protected static final int MAX_MSG_BYTE_LENGTH = 140;
/**
* The maximum amount of segments in the multipart message.
*/
protected static final int MAX_SEG_COUNT = 255;
private static final Logger LOG = LoggerFactory.getLogger(SmppSplitter.class);
/**
* Current reference number.
*/
private static int refNum;
private int messageLength;
private int segmentLength;
private int currentLength;
protected SmppSplitter(int messageLength, int segmentLength, int currentLength) {
this.messageLength = messageLength;
this.segmentLength = segmentLength;
this.currentLength = currentLength;
}
/**
* Returns reference number which length is
* {@link #UDHIE_SAR_REF_NUM_LENGTH}.
*
* @return the reference number of the multipart message
*/
protected static synchronized byte getReferenceNumber() {
refNum++;
if (refNum == 256) {
refNum = 1;
}
return (byte) refNum;
}
protected static synchronized byte getCurrentReferenceNumber() {
return (byte) refNum;
}
/**
* only needed for the unit tests
*/
protected static synchronized void resetCurrentReferenceNumber() {
SmppSplitter.refNum = 0;
}
public byte[][] split(byte[] message) {
if (!isSplitRequired()) {
return new byte[][] {message};
}
int segmentLength = getSegmentLength();
// determine how many messages
int segmentNum = message.length / segmentLength;
int messageLength = message.length;
if (segmentNum > MAX_SEG_COUNT) {
// this is too long, can't fit, so chop
segmentNum = MAX_SEG_COUNT;
messageLength = segmentNum * segmentLength;
}
if ((messageLength % segmentLength) > 0) {
segmentNum++;
}
byte[][] segments = new byte[segmentNum][];
int lengthOfData;
byte refNum = getReferenceNumber();
for (int i = 0; i < segmentNum; i++) {
LOG.trace("segment number = {}", i);
if (segmentNum - i == 1) {
lengthOfData = messageLength - i * segmentLength;
} else {
lengthOfData = segmentLength;
}
LOG.trace("Length of data = {}", lengthOfData);
segments[i] = new byte[UDHIE_HEADER_REAL_LENGTH + lengthOfData];
LOG.trace("segments[{}].length = {}", i, segments[i].length);
segments[i][0] = UDHIE_HEADER_LENGTH; // doesn't include itself, is header length
// SAR identifier
segments[i][1] = UDHIE_IDENTIFIER_SAR;
// SAR length
segments[i][2] = UDHIE_SAR_LENGTH;
// DATAGRAM REFERENCE NUMBER
segments[i][3] = refNum;
// total number of segments
segments[i][4] = (byte) segmentNum;
// segment #
segments[i][5] = (byte) (i + 1);
// now copy the data
System.arraycopy(message, i * segmentLength, segments[i], UDHIE_HEADER_REAL_LENGTH, lengthOfData);
}
return segments;
}
protected boolean isSplitRequired() {
return getCurrentLength() > getMessageLength();
}
/**
* Gets maximum message length.
*
* @return maximum message length
*/
public int getMessageLength() {
return messageLength;
}
/**
* Gets maximum segment length.
*
* @return maximum segment length
*/
public int getSegmentLength() {
return segmentLength;
}
/**
* Gets length of the message to split.
*
* @return length of the message to split
*/
public int getCurrentLength() {
return currentLength;
}
}