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

org.snmp4j.agent.BaseAgent Maven / Gradle / Ivy

/*_############################################################################
  _## 
  _##  SNMP4J-Agent 3 - BaseAgent.java  
  _## 
  _##  Copyright (C) 2005-2021  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.agent;

import java.io.*;

import org.snmp4j.*;
import org.snmp4j.cfg.EngineBootsCounterFile;
import org.snmp4j.agent.io.*;
import org.snmp4j.agent.mo.DefaultMOFactory;
import org.snmp4j.agent.mo.snmp.*;
import org.snmp4j.agent.mo.snmp4j.*;
import org.snmp4j.log.*;
import org.snmp4j.mp.*;
import org.snmp4j.security.*;
import org.snmp4j.smi.*;
import org.snmp4j.transport.*;

/**
 * The {@code BaseAgent} abstract class defines a framework for writing
 * SNMP agents using the SNMP4J-Agent API. To implement your own SNMP agent,
 * extend this class and implement the abstract methods defined by BaseAgent.
 * The hook methods do not need any specific implementation. They only provide
 * a defined mechanism to customize your agent.
 *
 * @author Frank Fock
 * @version 1.0
 * @deprecated Use {@link AgentConfigManager} instead. See {@link org.snmp4j.agent.example.SampleAgent} for sample
 * code on how to use {@link AgentConfigManager}. This class will be removed in SNMP4J-Agent 3.1.
 */
@Deprecated
public abstract class BaseAgent implements Runnable {

    private static final LogAdapter logger =
            LogFactory.getLogger(BaseAgent.class);

    public static final int STATE_CREATED = 0;
    public static final int STATE_INIT_STARTED = 10;
    public static final int STATE_INIT_FINISHED = 20;
    public static final int STATE_RUNNING = 40;
    public static final int STATE_STOPPED = 30;

    protected SNMPv2MIB snmpv2MIB;
    protected SnmpMpdMib snmpMpdMib;
    protected SnmpFrameworkMIB snmpFrameworkMIB;
    protected SnmpTargetMIB snmpTargetMIB;
    protected SnmpNotificationMIB snmpNotificationMIB;
    protected SnmpProxyMIB snmpProxyMIB;
    protected SnmpCommunityMIB snmpCommunityMIB;
    protected Snmp4jLogMib snmp4jLogMIB;
    protected Snmp4jConfigMib snmp4jConfigMIB;
    protected UsmMIB usmMIB;
    protected VacmMIB vacmMIB;
    protected DefaultMOServer server;
    protected Snmp session;
    protected TransportMapping[] transportMappings;
    protected MessageDispatcherImpl dispatcher;
    protected CommandProcessor agent;

    protected MPv3 mpv3;
    protected USM usm;

    protected EngineBootsCounterFile bootCounterFile;
    protected NotificationOriginator notificationOriginator;
    protected ProxyForwarder defaultProxyForwarder;

    protected OctetString sysDescr =
            new OctetString("SNMP4J-Agent - " +
                    System.getProperty("os.name", "") +
                    " - " + System.getProperty("os.arch") +
                    " - " + System.getProperty("os.version"));
    protected OID sysOID = new OID("1.3.6.1.4.1.4976");
    protected Integer32 sysServices = new Integer32(10);

    protected int agentState = STATE_CREATED;

    protected OctetString defaultContext;

    protected DefaultMOPersistenceProvider defaultPersistenceProvider;
    protected String configFileURI;

    /**
     * Creates a base agent with a {@link DefaultMOServer} as {@link MOServer}.
     * To use a different server implementation, modify the {@link #server} member
     * after construction.
     *
     * @param configURI
     *         the URI of the config file holding persistent data for this agent. If
     *         persistent data is not used then set this parameter to
     *         {@code null}.
     */
    protected BaseAgent(String configURI) {
        this.configFileURI = configURI;
        this.server = new DefaultMOServer();
        this.defaultPersistenceProvider =
                new DefaultMOPersistenceProvider(new MOServer[]{this.server},
                        configURI);
    }

    /**
     * Creates a base agent with boot-counter, config file, and a CommandProcessor
     * for processing SNMP requests.
     *
     * @param bootCounterFile
     *         a file with serialized boot-counter information (read/write). If the
     *         file does not exist it is created on shutdown of the agent.
     * @param configFile
     *         a file with serialized configuration information (read/write). If the
     *         file does not exist it is created on shutdown of the agent.
     * @param commandProcessor
     *         the {@code CommandProcessor} instance that handles the SNMP
     *         requests.
     */
    protected BaseAgent(File bootCounterFile,
                        File configFile,
                        CommandProcessor commandProcessor) {
        this((configFile == null) ? null : configFile.getPath());
        this.bootCounterFile = new EngineBootsCounterFile(bootCounterFile);
        this.agent = commandProcessor;
    }

    /**
     * Initialize transport mappings, message dispatcher, basic MIB modules,
     * proxy forwarder, VACM and USM security, and custom MIB modules and objects
     * provided by sub-classes.
     *
     * @throws IOException
     *         if initialization fails because transport initialization fails.
     */
    public void init() throws IOException {
        agentState = STATE_INIT_STARTED;
        initTransportMappings();
        initMessageDispatcher();
        server.addContext(new OctetString());
        snmpv2MIB = new SNMPv2MIB(sysDescr, sysOID, sysServices);
        snmpMpdMib = new SnmpMpdMib(DefaultMOFactory.getInstance());

        // register Snmp counters for updates
        dispatcher.addCounterListener(snmpv2MIB);
        dispatcher.addCounterListener(snmpMpdMib);
        mpv3.getCounterSupport().addCounterListener(snmpMpdMib);
        agent.addCounterListener(snmpv2MIB);
        agent.addCounterListener(snmpMpdMib);
        snmpFrameworkMIB =
                new SnmpFrameworkMIB(usm.getLocalEngineID(), (USM)
                        mpv3.getSecurityModel(SecurityModel.SECURITY_MODEL_USM),
                        dispatcher.getTransportMappings());
        usmMIB = new UsmMIB(usm, SecurityProtocols.getInstance());
        usm.addUsmUserListener(usmMIB);

        vacmMIB = new VacmMIB(new MOServer[]{server});
        snmpTargetMIB = new SnmpTargetMIB(dispatcher);
        snmpNotificationMIB = new SnmpNotificationMIB();
        snmpCommunityMIB = new SnmpCommunityMIB(snmpTargetMIB);
        initConfigMIB();
        snmpProxyMIB = new SnmpProxyMIB();
        notificationOriginator =
                new NotificationOriginatorImpl(session, vacmMIB,
                        snmpv2MIB.getSysUpTime(),
                        snmpTargetMIB, snmpNotificationMIB, snmpCommunityMIB);
        snmpv2MIB.setNotificationOriginator(agent);
        usm.getCounterSupport().addCounterListener(snmpv2MIB);

        setupDefaultProxyForwarder();
        // add USM users
        addUsmUser(usm);
        // add SNMPv1/v2c community to SNMPv3 security name mappings
        addCommunities(snmpCommunityMIB);
        addViews(vacmMIB);
        addNotificationTargets(snmpTargetMIB, snmpNotificationMIB);

        registerSnmpMIBs();
    }

    protected void initConfigMIB() {
        snmp4jLogMIB = new Snmp4jLogMib();
        if ((configFileURI != null) && (defaultPersistenceProvider != null)) {
            snmp4jConfigMIB = new Snmp4jConfigMib(snmpv2MIB.getSysUpTime());
            snmp4jConfigMIB.setSnmpCommunityMIB(snmpCommunityMIB);
            snmp4jConfigMIB.setPrimaryProvider(defaultPersistenceProvider);
        }
    }

    /**
     * This method can be overwritten by a subagent to specify the contexts
     * each MIB module (group) will be registered to.
     *
     * @param mibGroup
     *         a group of {@link ManagedObject}s (i.e., a MIB module).
     *
     * @return the context for which the module should be registered.
     * @since 1.1
     */
    protected OctetString getContext(MOGroup mibGroup) {
        return getDefaultContext();
    }

    /**
     * Register the basic MIB modules at the agent's {@code MOServer}.
     */
    protected void registerSnmpMIBs() {
        try {
            snmpTargetMIB.registerMOs(server, getContext(snmpTargetMIB));
            snmpNotificationMIB.registerMOs(server, getContext(snmpNotificationMIB));
            vacmMIB.registerMOs(server, getContext(vacmMIB));
            usmMIB.registerMOs(server, getContext(usmMIB));
            snmpv2MIB.registerMOs(server, getContext(snmpv2MIB));
            snmpMpdMib.registerMOs(server, getContext(snmpMpdMib));
            snmpFrameworkMIB.registerMOs(server, getContext(snmpFrameworkMIB));
            snmpCommunityMIB.registerMOs(server, getContext(snmpCommunityMIB));
            snmp4jLogMIB.registerMOs(server, getContext(snmp4jLogMIB));
            if (snmp4jConfigMIB != null) {
                snmp4jConfigMIB.registerMOs(server, getContext(snmp4jConfigMIB));
            }
            snmpProxyMIB.registerMOs(server, getContext(snmpProxyMIB));
            registerManagedObjects();
        } catch (DuplicateRegistrationException ex) {
            ex.printStackTrace();
        }
    }

    /**
     * Unregister the basic MIB modules from the agent's {@code MOServer}.
     */
    protected void unregisterSnmpMIBs() {
        snmpTargetMIB.unregisterMOs(server, getContext(snmpTargetMIB));
        snmpNotificationMIB.unregisterMOs(server, getContext(snmpNotificationMIB));
        vacmMIB.unregisterMOs(server, getContext(vacmMIB));
        usmMIB.unregisterMOs(server, getContext(usmMIB));
        snmpv2MIB.unregisterMOs(server, getContext(snmpv2MIB));
        snmpFrameworkMIB.unregisterMOs(server, getContext(snmpFrameworkMIB));
        snmpCommunityMIB.unregisterMOs(server, getContext(snmpCommunityMIB));
        snmp4jLogMIB.unregisterMOs(server, getContext(snmp4jLogMIB));
        if (snmp4jConfigMIB != null) {
            snmp4jConfigMIB.unregisterMOs(server, getContext(snmp4jConfigMIB));
        }
        snmpProxyMIB.unregisterMOs(server, getContext(snmpProxyMIB));
        unregisterManagedObjects();
    }

    /**
     * Register additional managed objects at the agent's server.
     */
    protected abstract void registerManagedObjects();

    /**
     * Unregister additional managed objects from the agent's server.
     */
    protected abstract void unregisterManagedObjects();

    /**
     * Creates and registers the default proxy forwarder application
     * ({@link ProxyForwarderImpl}).
     */
    protected void setupDefaultProxyForwarder() {
        defaultProxyForwarder = new ProxyForwarderImpl(session, snmpProxyMIB,
                snmpTargetMIB);
        agent.addProxyForwarder(defaultProxyForwarder,
                null, ProxyForwarder.PROXY_TYPE_ALL);
        ((ProxyForwarderImpl) defaultProxyForwarder).addCounterListener(snmpv2MIB);
    }

    /**
     * Loads the configuration using the specified import mode from the set
     * config file.
     *
     * @param importMode
     *         one of the import modes defined by {@link ImportMode}.
     */
    public void loadConfig(int importMode) {
        try {
            defaultPersistenceProvider.restore(null, importMode);
        } catch (IOException ex) {
            logger.error(ex);
        }
    }

    /**
     * Save the current (serializable) managed object configuration into
     * the config file.
     */
    public void saveConfig() {
        try {
            defaultPersistenceProvider.store(configFileURI, snmp4jConfigMIB);
        } catch (IOException ex) {
            logger.error(ex);
        }
    }

    /**
     * Adds a shutdown hook that saves the internal config into the config file
     * when a SIGTERM (Ctrl-C) is terminating the agent.
     */
    protected void addShutdownHook() {
        Runtime.getRuntime().addShutdownHook(new Thread() {
            public void run() {
                saveConfig();
            }
        });
    }

    /**
     * Finishes initialization of the agent by connecting server and command
     * processor, setting up USM, VACM, notification targets, and finally sending
     * a coldStart notification to configured targets.
     */
    protected void finishInit() {
        if (agentState < STATE_INIT_STARTED) {
            logger.fatal("Agent initialization finish is called before " +
                    "initialization, current state is " + agentState);
        }
        agent.setNotificationOriginator(notificationOriginator);
        agent.addMOServer(server);
        agent.setCoexistenceProvider(snmpCommunityMIB);
        agent.setVacm(vacmMIB);
        dispatcher.addCommandResponder(agent);
        agentState = STATE_INIT_FINISHED;
    }

    protected void sendColdStartNotification() {
        notificationOriginator.notify(new OctetString(), SnmpConstants.coldStart,
                new VariableBinding[0]);
    }

    /**
     * Starts the agent by let it listen on the configured SNMP agent ports
     * (transpot mappings).
     */
    public void run() {
        if (agentState < STATE_INIT_FINISHED) {
            logger.fatal("Agent is run uninitialized, current state is " + agentState);
        }
        if (agentState == STATE_STOPPED) {
            initSnmpSession();
        }
        try {
            session.listen();
            agentState = STATE_RUNNING;
        } catch (IOException iox) {
            logger.error(iox);
        }
    }

    /**
     * Stops the agent by closing the SNMP session and associated transport
     * mappings.
     *
     * @since 1.1
     */
    public void stop() {
        if (agentState != STATE_RUNNING) {
            logger.error("Agent is stopped although it is not running, " +
                    "current state is " + agentState);
        }
        try {
            session.close();
        } catch (IOException ex) {
            logger.warn("Closing agent session threw IOException: " + ex.getMessage());
        }
        session = null;
        agentState = STATE_STOPPED;
    }

    /**
     * Initializes the message dispatcher ({@link MessageDispatcherImpl}) with
     * the transport mappings.
     */
    protected void initMessageDispatcher() {
        dispatcher = new MessageDispatcherImpl();
        mpv3 = new MPv3(agent.getContextEngineID().getValue());
        usm = new USM(SecurityProtocols.getInstance(),
                agent.getContextEngineID(),
                updateEngineBoots());
        SecurityModels.getInstance().addSecurityModel(usm);
        SecurityProtocols.getInstance().addDefaultProtocols();
        dispatcher.addMessageProcessingModel(new MPv1());
        dispatcher.addMessageProcessingModel(new MPv2c());
        dispatcher.addMessageProcessingModel(mpv3);
        initSnmpSession();
    }

    protected void initSnmpSession() {
        session = new Snmp(dispatcher);
        for (TransportMapping transportMapping : transportMappings) {
            try {
                session.addTransportMapping(transportMapping);
            } catch (Exception ex) {
                logger.warn("Failed to initialize transport mapping '" +
                        transportMapping + "' with: " + ex.getMessage());
            }
        }
        updateSession(session);
    }

    /**
     * Updates all objects with a new session instance. This method must be
     * overwritten, if non-default SNMP MIB instances are created by a subclass.
     *
     * @param session
     *         a SNMP Session instance.
     */
    protected void updateSession(Session session) {
        if (notificationOriginator instanceof NotificationOriginatorImpl) {
            ((NotificationOriginatorImpl) notificationOriginator).setSession(session);
        }
        if (defaultProxyForwarder instanceof ProxyForwarderImpl) {
            ((ProxyForwarderImpl) defaultProxyForwarder).setSession(session);
        }
    }

    /**
     * Updates the engine boots counter and returns the actual value.
     *
     * @return the actual boots counter value.
     */
    protected int updateEngineBoots() {
        return bootCounterFile.updateEngineBoots();
    }

    /**
     * Reads the engine boots counter from the corresponding input stream (file).
     *
     * @return the boots counter value read or zero if it could not be read.
     */
    protected int getEngineBoots() {
        return bootCounterFile.getEngineBoots();
    }

    /**
     * Adds all the necessary initial users to the USM.
     *
     * @param usm
     *         the USM instance used by this agent.
     */
    protected abstract void addUsmUser(USM usm);

    /**
     * Adds initial notification targets and filters.
     *
     * @param targetMIB
     *         the SnmpTargetMIB holding the target configuration.
     * @param notificationMIB
     *         the SnmpNotificationMIB holding the notification (filter)
     *         configuration.
     */
    protected abstract void addNotificationTargets(SnmpTargetMIB targetMIB,
                                                   SnmpNotificationMIB notificationMIB);

    /**
     * Adds initial VACM configuration.
     *
     * @param vacmMIB
     *         the VacmMIB holding the agent's view configuration.
     */
    protected abstract void addViews(VacmMIB vacmMIB);

    /**
     * Adds community to security name mappings needed for SNMPv1 and SNMPv2c.
     *
     * @param communityMIB
     *         the SnmpCommunityMIB holding coexistence configuration for community
     *         based security models.
     */
    protected abstract void addCommunities(SnmpCommunityMIB communityMIB);

    /**
     * Initializes the transport mappings (ports) to be used by the agent.
     *
     * @throws IOException
     *         if an IO exception occurs while initializing the default transport mapping on all local IP addresses on
     *         port 161.
     */
    protected void initTransportMappings() throws IOException {
        transportMappings = new DefaultUdpTransportMapping[]{
                new DefaultUdpTransportMapping(new UdpAddress("0.0.0.0/161"))};
    }

    public NotificationOriginator getNotificationOriginator() {
        return notificationOriginator;
    }

    public void setDefaultProxyForwarder(ProxyForwarder defaultProxyForwarder) {
        this.defaultProxyForwarder = defaultProxyForwarder;
    }

    public void setSysDescr(OctetString sysDescr) {
        this.sysDescr.setValue(sysDescr.getValue());
    }

    public void setSysOID(OID sysOID) {
        this.sysOID.setValue(sysOID.getValue());
    }

    public void setSysServices(Integer32 sysServices) {
        this.sysServices.setValue(sysServices.getValue());
    }

    public void setAgent(CommandProcessor agent) {
        this.agent = agent;
    }

    public void setBootCounterFile(File bootCounterFile) {
        this.bootCounterFile = new EngineBootsCounterFile(bootCounterFile);
    }

    public void setConfigFile(File configFile) {
        this.configFileURI = configFile.getPath();
    }

    /**
     * Sets the default context for this base agent. By setting this value before
     * any MIB modules have been registered at the internal server, the context
     * for which the registration is performed can be changed. The default context
     * is {@code null} which causes MIB objects to be virtually registered
     * for all contexts.
     *
     * @param defaultContext
     *         the context for default MIB objects.
     *
     * @since 1.1
     */
    public void setDefaultContext(OctetString defaultContext) {
        this.defaultContext = defaultContext;
    }

    public ProxyForwarder getDefaultProxyForwarder() {
        return defaultProxyForwarder;
    }

    public OctetString getSysDescr() {
        return sysDescr;
    }

    public OID getSysOID() {
        return sysOID;
    }

    public Integer32 getSysServices() {
        return sysServices;
    }

    public CommandProcessor getAgent() {
        return agent;
    }

    public File getBootCounterFile() {
        return bootCounterFile.getBootCounterFile();
    }

    public File getConfigFile() {
        return new File(configFileURI);
    }

    public Snmp4jConfigMib getSnmp4jConfigMIB() {
        return snmp4jConfigMIB;
    }

    public Snmp4jLogMib getSnmp4jLogMIB() {
        return snmp4jLogMIB;
    }

    public SnmpCommunityMIB getSnmpCommunityMIB() {
        return snmpCommunityMIB;
    }

    public SnmpFrameworkMIB getSnmpFrameworkMIB() {
        return snmpFrameworkMIB;
    }

    public SnmpNotificationMIB getSnmpNotificationMIB() {
        return snmpNotificationMIB;
    }

    public SnmpProxyMIB getSnmpProxyMIB() {
        return snmpProxyMIB;
    }

    public SnmpTargetMIB getSnmpTargetMIB() {
        return snmpTargetMIB;
    }

    public SNMPv2MIB getSnmpv2MIB() {
        return snmpv2MIB;
    }

    public SnmpMpdMib getSnmpMpdMib() {
        return snmpMpdMib;
    }

    public UsmMIB getUsmMIB() {
        return usmMIB;
    }

    public VacmMIB getVacmMIB() {
        return vacmMIB;
    }

    public Snmp getSession() {
        return session;
    }

    public DefaultMOServer getServer() {
        return server;
    }

    public MPv3 getMPv3() {
        return mpv3;
    }

    public USM getUsm() {
        return usm;
    }

    /**
     * Returns the agent's state.
     *
     * @return one of the state's starting from {@link #STATE_CREATED} to
     * {@link #STATE_RUNNING}.
     * @since 1.1
     */
    public int getAgentState() {
        return agentState;
    }

    /**
     * Returns the default context - which is the context that is used by the
     * base agent to register its MIB objects. By default it is {@code null}
     * which causes the objects to be registered virtually for all contexts.
     * In that case, subagents for example my not register their own objects
     * under the same subtree(s) in any context. To allow subagents to register
     * their own instances of those MIB modules, an empty {@code OctetString}
     * should be used as default context instead.
     *
     * @return {@code null} or an {@code OctetString} (normally the empty
     * string) denoting the context used for registering default MIBs.
     * @since 1.1
     */
    public OctetString getDefaultContext() {
        return defaultContext;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy