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

org.snmp4j.fluent.SnmpCompletableFuture Maven / Gradle / Ivy

There is a newer version: 3.7.8_1
Show newest version
/*_############################################################################
  _## 
  _##  SNMP4J - SnmpCompletableFuture.java  
  _## 
  _##  Copyright (C) 2003-2022  Frank Fock (SNMP4J.org)
  _##  
  _##  Licensed 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.snmp4j.fluent;

import org.snmp4j.*;
import org.snmp4j.event.ResponseEvent;
import org.snmp4j.event.ResponseListener;
import org.snmp4j.smi.Address;

import java.io.IOException;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.TimeoutException;

/**
 * The {@link SnmpCompletableFuture} implements a {@link CompletableFuture} for a {@link PDU}. It represents
 * the result of an SNMP operation sending a {@link PDU} and receiving its response when the sent {@link PDU} was a
 * confirmed one.
 * This future can be completed normally and exceptionally. A normal completion is a successful SNMP operation.
 * An unconfirmed operation completes always normally if the sending of the SNMP message was successful.
 * Any error situation, like a response {@link PDU} with a non-zero error status, timeout, or {@link PDU#REPORT}
 * PDU response. In the latter case, a {@link ReportException} will be returned. For a timeout,
 * a {@link TimeoutException}, and for a SNMP error, a {@link SnmpErrorException}.
 * If sending the SNMP message throws an exception, this exception is directly returned.
 *
 * @author Frank Fock
 * @since 3.5.0
 */
public class SnmpCompletableFuture extends CompletableFuture implements ResponseListener {

    public static final String ERR_TIMEOUT = "SNMP Request timed out";

    private ResponseEvent responseEvent;
    private Object[] userObjects;

    protected String timeoutMessage = ERR_TIMEOUT;

    protected SnmpCompletableFuture() {
    }

    @Override
    public  void onResponse(ResponseEvent event) {
        ((Session) event.getSource()).cancel(event.getRequest(), this);
        this.responseEvent = event;
        this.userObjects = (Object[])event.getUserObject();
        if (event.getError() != null) {
            completeExceptionally(event.getError());
        } else if (event.getResponse() == null) {
            completeExceptionally(new TimeoutException(timeoutMessage));
        } else if (event.getResponse().getType() == PDU.REPORT) {
            completeExceptionally(new ReportException((ScopedPDU) event.getResponse()));
        } else if (event.getResponse().getErrorStatus() != PDU.noError) {
            completeExceptionally(new SnmpErrorException(
                    event.getResponse().getErrorStatus(),
                    event.getResponse().getErrorIndex()));
        } else {
            complete(event.getResponse());
        }
    }

    public ResponseEvent getResponseEvent() {
        return responseEvent;
    }

    /**
     * Sedn the given {@link PDU} using the provider {@link Snmp} session to the specified {@link Target}.
     * @param snmp
     *    a {@link Snmp} instance, preferrably created by {@link SnmpBuilder}.
     * @param target
     *    the target where to send the message to with pdu as payload.
     * @param pdu
     *    the PDU to send.
     * @param userObjects
     *    a list of user objects to return with this {@link CompletableFuture} by calling {@link #getUserObjects()}.
     * @return
     *    a new {@link SnmpCompletableFuture} that will receive the result of this operation.
     * @param 
     *    the address type.
     */
    public static  SnmpCompletableFuture send(Snmp snmp, Target target, PDU pdu,
                                                                 Object... userObjects) {
        SnmpCompletableFuture future = new SnmpCompletableFuture();
        try {
            snmp.send(pdu, target, userObjects, future);
            if (!pdu.isConfirmedPdu()) {
                future.complete(null);
            }
        }
        catch (IOException iox) {
            future.completeExceptionally(iox);
        }
        return future;
    }

    /**
     * Returns the user objects provided by {@link #send(Snmp, Target, PDU, Object...)} when this future completed.
     * @return
     *    a list of user objects.
     */
    public Object[] getUserObjects() {
        return userObjects;
    }

    /**
     * This {@code ReportException} represents an SNMP {@link PDU#REPORT} PDU.
     */
    public static class ReportException extends Exception {
        private static final long serialVersionUID = 1539501546791678999L;
        private final ScopedPDU report;

        public ReportException(ScopedPDU report) {
            super("SNMP REPORT error: "+report);
            this.report = report;
        }

        public ScopedPDU getReport() {
            return report;
        }
    }

    /**
     * This {@code SnmpErrorException} represents an SNMP error status and index.
     */
    public static class SnmpErrorException extends Exception {
        private static final long serialVersionUID = -2338397456373295345L;
        private final int errorStatus;
        private final int errorIndex;

        public SnmpErrorException(int errorStatus, int errorIndex) {
            super("SNMP error "+PDU.toErrorStatusText(errorStatus)+ " on index "+errorIndex);
            this.errorStatus = errorStatus;
            this.errorIndex = errorIndex;
        }

        public int getErrorStatus() {
            return errorStatus;
        }

        public int getErrorIndex() {
            return errorIndex;
        }

        public String getErrorMessage() {
            return PDU.toErrorStatusText(errorStatus);
        }

    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy