TestTCompactProtocol.java

/*
 * 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.thrift.protocol;

import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertThrows;

import java.util.Arrays;
import org.apache.thrift.TDeserializer;
import org.apache.thrift.TException;
import org.apache.thrift.transport.TMemoryInputTransport;
import org.junit.jupiter.api.Test;
import thrift.test.Bonk;

public class TestTCompactProtocol extends ProtocolTestBase {
  @Override
  protected TProtocolFactory getFactory() {
    return new TCompactProtocol.Factory();
  }

  @Override
  protected boolean canBeUsedNaked() {
    return true;
  }

  @Test
  public void testOOMDenialOfService() throws Exception {
    // Struct header, Integer.MAX_VALUE length, and only one real
    // byte of data
    byte[] bytes = {24, -1, -1, -1, -17, 49};
    TDeserializer deser = new TDeserializer(new TCompactProtocol.Factory(1000));
    Bonk bonk = new Bonk();
    try {
      deser.deserialize(bonk, bytes);
    } catch (TException e) {
      // Ignore as we are only checking for OOM in the failure case
    }
  }

  @Test
  public void testReadVarint64FastPathRejectsOverlong() throws Exception {
    byte[] buf = new byte[11];
    Arrays.fill(buf, (byte) 0x80); // 11 continuation bytes, no terminator
    TCompactProtocol proto = new TCompactProtocol(new TMemoryInputTransport(buf));
    TProtocolException ex = assertThrows(TProtocolException.class, proto::readI64);
    assertEquals(TProtocolException.INVALID_DATA, ex.getType());
  }

  @Test
  public void testReadVarint64FastPathAcceptsValid10Byte() throws Exception {
    byte[] buf = new byte[10];
    Arrays.fill(buf, (byte) 0x80);
    buf[9] = 0x01; // terminating byte
    TCompactProtocol proto = new TCompactProtocol(new TMemoryInputTransport(buf));
    assertDoesNotThrow(proto::readI64);
  }

  @Test
  public void testReadVarint32FastPathRejectsOverlong() throws Exception {
    byte[] buf = new byte[6];
    Arrays.fill(buf, (byte) 0x80); // 6 continuation bytes, no terminator
    TCompactProtocol proto = new TCompactProtocol(new TMemoryInputTransport(buf));
    TProtocolException ex = assertThrows(TProtocolException.class, proto::readI32);
    assertEquals(TProtocolException.INVALID_DATA, ex.getType());
  }

  @Test
  public void testReadVarint32FastPathAcceptsValid5Byte() throws Exception {
    byte[] buf = new byte[5];
    Arrays.fill(buf, (byte) 0x80);
    buf[4] = 0x01; // terminating byte
    TCompactProtocol proto = new TCompactProtocol(new TMemoryInputTransport(buf));
    assertDoesNotThrow(proto::readI32);
  }

  public static void main(String args[]) throws Exception {
    new TestTCompactProtocol().benchmark();
  }
}