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

com.caucho.server.snmp.SnmpParser Maven / Gradle / Ivy

/*
 * Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
 *
 * This file is part of Resin(R) Open Source
 *
 * Each copy or derived work must preserve the copyright notice and this
 * notice unmodified.
 *
 * Resin Open Source is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * Resin Open Source 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, or any warranty
 * of NON-INFRINGEMENT.  See the GNU General Public License for more
 * details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Resin Open Source; if not, write to the
 *
 *   Free Software Foundation, Inc.
 *   59 Temple Place, Suite 330
 *   Boston, MA 02111-1307  USA
 *
 * @author Nam Nguyen
 */

package com.caucho.server.snmp;

import java.io.IOException;
import java.io.InputStream;

import com.caucho.server.snmp.types.*;
import com.caucho.util.L10N;

public class SnmpParser
{
  private static final L10N L = new L10N(SnmpParser.class);

  private InputStream _is;
  private int _bytesRead = 0;
  
  private boolean _isCheckIdentifier = true;
  
  public SnmpParser(InputStream is) 
  {
    _is = is;
  }
  
  public SnmpParser(String s)
  {
    _is = new StringInputStream(s);
  }
  
  public SnmpMessageValue readMessage()
    throws IOException
  {
    checkIdentifier(SnmpValue.SEQUENCE);

    int expectedLen = readLength();
    int bytesRead = getBytesRead();
    
    IntegerValue version = readInteger();
    OctetStringValue communityString = readOctetString();

    int id = readByte();
    _isCheckIdentifier = false;
    
    PduValue pdu;
    switch(id) {
      case SnmpValue.GET_REQUEST_PDU:
        pdu = readGetRequestPdu();
        break;
      case SnmpValue.GET_NEXT_REQUEST_PDU:
        pdu = readGetNextRequestPdu();
        break;
      case SnmpValue.SET_REQUEST_PDU:
        pdu = readSetRequestPdu();
        break;
      case SnmpValue.GET_RESPONSE_PDU:
        pdu = readGetResponsePdu();
        break;
      //case SnmpValue.TRAP_PDU:
        //break;
      default:
        throw new SnmpParsingException(L.l("unknown PDU type '{0}'", id));
    }
    
    checkLength(expectedLen, getBytesRead() - bytesRead);
    
    return new SnmpMessageValue(version, communityString, pdu);
  }

  public PduValue readGetRequestPdu()
    throws IOException
  {
    return readPdu(SnmpValue.GET_REQUEST_PDU);
  }
  
  public PduValue readGetNextRequestPdu()
    throws IOException
  {
    return readPdu(SnmpValue.GET_NEXT_REQUEST_PDU);
  }
  
  public PduValue readSetRequestPdu()
    throws IOException
  {
    return readPdu(SnmpValue.SET_REQUEST_PDU);
  }
  
  public PduValue readGetResponsePdu()
    throws IOException
  {
    return readPdu(SnmpValue.GET_RESPONSE_PDU);
  }
  
  private PduValue readPdu(int type)
    throws IOException
  {
    checkIdentifier(type);
    
    int expectedLen = readLength();
    int bytesRead = getBytesRead();
    
    IntegerValue requestId = readInteger();
    
    // errors are ignored for requests
    IntegerValue error = readInteger();
    IntegerValue errorIndex = readInteger();
    
    VarBindListValue varBindList = readVarBindList();
    
    checkLength(expectedLen, getBytesRead() - bytesRead);
    
    return PduValue.create(type,
                           requestId,
                           error,
                           errorIndex,
                           varBindList);
  }

  public VarBindListValue readVarBindList()
     throws IOException
  {
    checkIdentifier(SnmpValue.SEQUENCE);
    
    int expectedLen = readLength();
    int bytesRead = 0;
    
    VarBindListValue varBindList = new VarBindListValue();
    
    while (bytesRead < expectedLen) {
      bytesRead -= getBytesRead();
      
      varBindList.addVarBind(readVarBind());
      
      bytesRead += getBytesRead();
    }
    
    checkLength(expectedLen, bytesRead);
    
    return varBindList;
  }
  
  public VarBindValue readVarBind()
    throws IOException
  {
    checkIdentifier(SnmpValue.SEQUENCE);
    
    int expectedLen = readLength();
    int bytesRead = getBytesRead();
    
    VarBindValue varBind = new VarBindValue(readObjectIdentifier(),
                                            read());

    checkLength(expectedLen, getBytesRead() - bytesRead);
    
    return varBind;
  }
  
  public void skipObject()
    throws IOException
  {
    readByte();
    
    int len = readLength();
    
    for (int i = 0; i < len; i++) {
      readByte();
    }
  }
  
  public SnmpValue read()
    throws IOException
  {
    int identifier = readByte();
    
    _isCheckIdentifier = false;

    switch (identifier) {
      case SnmpValue.NULL:
        return readNull();
      case SnmpValue.INTEGER:
        return readInteger();
      case SnmpValue.OCTET_STRING:
        return readOctetString();
      case SnmpValue.OBJECT_IDENTIFIER:
        return readObjectIdentifier();
      case SnmpValue.SEQUENCE:
        return readSequence();
      case SnmpValue.IP_ADDRESS:
        return readIpAddress();
      case SnmpValue.COUNTER:
        return readCounter();
      case SnmpValue.GAUGE:
        return readGauge();
      case SnmpValue.TIME_TICKS:
        return readTimeTicks();
      case SnmpValue.OPAQUE:
        return readOpaque();
      case SnmpValue.GET_REQUEST_PDU:
        return readGetRequestPdu();
      case SnmpValue.GET_NEXT_REQUEST_PDU:
        return readGetNextRequestPdu();
      case SnmpValue.GET_RESPONSE_PDU:
        return readGetResponsePdu();
      case SnmpValue.SET_REQUEST_PDU:
        return readSetRequestPdu();
      //case SnmpValue.TRAP_PDU:
      //  break;
      default:
        throw new SnmpParsingException(L.l("unknown identifier {0}", identifier));
    }
  }
  
  
  public NullValue readNull()
    throws IOException
  {
    checkIdentifier(SnmpValue.NULL);
    int len = readLength();

    if (len != 0)
      throw new SnmpParsingException("length must be zero for NULL");
    
    return NullValue.NULL;
  }
  
  public IntegerValue readInteger()
    throws IOException
  {
    checkIdentifier(SnmpValue.INTEGER);
    int len = readLength();

    if (len < 1 || len > 4)
      throw new SnmpParsingException(L.l("integer length {0} must be 1-4"));

    int value =  readByte();
    boolean isNeg = (value & 0x80) > 0;

    for (int i = 1; i < len; i++) {
      value <<= 8;
      value |= readByte();
    }
    
    if (isNeg) {
      while (len != 4) {
        value |= 0xFF << (8 * len++);
      }
    }
    
    return new IntegerValue(value);
  }

  public OctetStringValue readOctetString()
    throws IOException
  {
    checkIdentifier(SnmpValue.OCTET_STRING);
    int expectedLen = readLength();
    
    StringBuilder sb = new StringBuilder();
    
    for (int i = 0; i < expectedLen; i++) {
      sb.append((char) readByte());
    }

    return new OctetStringValue(sb.toString());
  }
  
  public ObjectIdentifierValue readObjectIdentifier()
    throws IOException
  {
    checkIdentifier(SnmpValue.OBJECT_IDENTIFIER);
    int len = readLength();
    
    StringBuilder sb = new StringBuilder();

    int total = 0;
    while (len-- > 0) {
      int b = readByte();
      
      total <<= 7;
      total += (b & 0x7F);
      
      if ((b & 0x80) == 0) {
        break;
      }
    }
    
    // total is (40x + y)
    // x can only be 0,1,2
    // y can not be more than 39 for x = 0,1
    
    int x;
    int y;
    
    if (total < 40) {
      x = 0;
      y = total;
    }
    else if (total < 80) {
      x = 1;
      y = total - 40;
    }
    else {
      x = 2;
      y = total - 80;
    }
    
    sb.append(x);
    sb.append('.');
    sb.append(y);
    
    while (len > 0) {
      int val = 0;
      
      while (len-- > 0) {
        int b = readByte();
        
        val <<= 7;
        val += (b & 0x7F);
        
        if ((b & 0x80) == 0) {
          break;
        }
      }
      
      sb.append('.');
      sb.append(val);
    }

    return new ObjectIdentifierValue(sb.toString());
  }
  
  public IpAddressValue readIpAddress()
    throws IOException
  {
    checkIdentifier(SnmpValue.IP_ADDRESS);
    int expectedLen = readLength();
    
    StringBuilder sb = new StringBuilder();
    
    for (int i = 0; i < expectedLen; i++) {
      int b = readByte();
      
      if (b < 0x00 || b > 0xFF) {
        throw new SnmpParsingException(L.l("IP address digit {0} out of range", b));
      }
      
      if (i != 0)
        sb.append('.');

      sb.append(b);
    }

    return new IpAddressValue(sb.toString());
  }
  
  public CounterValue readCounter()
    throws IOException
  {
    checkIdentifier(SnmpValue.COUNTER);
    int expectedLen = readLength();
    
    long value = 0;
    
    for (int i = 0; i < expectedLen; i++) {
      value <<= 8;
      value |= readByte();
    }

    return new CounterValue(value);
  }
  
  public GaugeValue readGauge()
    throws IOException
  {
    checkIdentifier(SnmpValue.GAUGE);
    int expectedLen = readLength();
    
    long value = 0;
    
    for (int i = 0; i < expectedLen; i++) {
      value <<= 8;
      value |= readByte();
    }

    return new GaugeValue(value);
  }
  
  public TimeTicksValue readTimeTicks()
    throws IOException
  {
    checkIdentifier(SnmpValue.TIME_TICKS);
    int expectedLen = readLength();
    
    long value = 0;
    
    for (int i = 0; i < expectedLen; i++) {
      value <<= 8;
      value |= readByte();
    }

    return new TimeTicksValue(value);
  }
  
  public OpaqueValue readOpaque()
    throws IOException
  {
    checkIdentifier(SnmpValue.OPAQUE);
    int expectedLen = readLength();
    
    StringBuilder sb = new StringBuilder();
    
    for (int i = 0; i < expectedLen; i++) {
      sb.append((char) readByte());
    }

    return new OpaqueValue(sb.toString());
  }

  public SequenceValue readSequence()
    throws IOException
  {
    checkIdentifier(SnmpValue.SEQUENCE);
    int expectedLen = readLength();
    
    SequenceValue sequence = new SequenceValue();
    
    int bytesRead = 0;
    while (bytesRead < expectedLen) {
      bytesRead -= getBytesRead();
      
      SnmpValue item = read();
      sequence.add(item);

      bytesRead += getBytesRead();
    }

    if (bytesRead != expectedLen) {
      throw new SnmpParsingException(L.l("expected sequence length {0} != {1}",
                                          expectedLen, bytesRead));
    }

    return sequence;
  }

  public int readLength()
    throws IOException
  {
    int octlet = readByte();

    if ((octlet & 0x80) == 0)
      return octlet & 0x7F;
    
    int numOfOctlets = octlet & 0x7F;

    int len = 0;
    for (int i = 0; i < numOfOctlets; i++) {
      octlet = readByte();
      
      len <<= 8;
      len |= octlet;
    }

    return len;
  }
  
  private int readByte()
    throws IOException
  {
    _bytesRead++;
    
    int ch = _is.read();
    
    if (ch < 0)
      throw new SnmpParsingException("unexpected EOF");
    
    return ch;
  }

  private int getBytesRead()
  {
    return _bytesRead;
  }
  
  private void checkLength(int expected, int bytesRead)
    throws SnmpParsingException
  {
    if (expected != bytesRead) {
      throw new SnmpParsingException(L.l("expected length {0} != {1}", expected, bytesRead));
    }
  }
  
  private void checkIdentifier(int expected)
    throws IOException
  {
    if (_isCheckIdentifier)
      checkIdentifier(expected, readByte());
    
    _isCheckIdentifier = true;
  }
  
  private void checkIdentifier(int expected, int identifier)
    throws SnmpParsingException
  {
    if ( expected != identifier) {
      throw new SnmpParsingException(L.l("saw '{0}' but expected type '{1}'", type(identifier), type(expected)));
    }
  }
  
  private static String type(int identifier)
  {
    return SnmpValue.typeName(identifier);
  }
  
  static class StringInputStream extends InputStream
  {
    private String _s;
    private int _len;
    private int _index;
    
    public StringInputStream(String s)
    {
      _s = s;
      _len = s.length();
    }
    
    public int read()
      throws IOException
    {
      if (_index < _len)
        return _s.charAt(_index++);
      else
        return -1;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy