/******************************************************************************* * This file is part of OpenNMS(R). * * Copyright (C) 2006-2011 The OpenNMS Group, Inc. * OpenNMS(R) is Copyright (C) 1999-2011 The OpenNMS Group, Inc. * * OpenNMS(R) is a registered trademark of The OpenNMS Group, Inc. * * OpenNMS(R) 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 3 of the License, * or (at your option) any later version. * * OpenNMS(R) 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. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with OpenNMS(R). If not, see: * http://www.gnu.org/licenses/ * * For more information contact: * OpenNMS(R) Licensing <license@opennms.org> * http://www.opennms.org/ * http://www.opennms.com/ *******************************************************************************/ package org.opennms.netmgt.syslogd; import static org.junit.Assert.assertEquals; import static org.opennms.core.utils.InetAddressUtils.addr; import java.io.ByteArrayInputStream; import java.io.InputStream; import java.lang.reflect.UndeclaredThrowableException; import java.net.BindException; import java.net.DatagramPacket; import java.net.UnknownHostException; import java.util.Arrays; import java.util.List; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import org.apache.commons.io.IOUtils; import org.apache.log4j.Level; import org.junit.After; import org.junit.Before; import org.junit.Ignore; import org.junit.Test; import org.junit.runner.RunWith; import org.opennms.core.concurrent.WaterfallExecutor; import org.opennms.core.test.MockLogAppender; import org.opennms.core.test.OpenNMSJUnit4ClassRunner; import org.opennms.core.utils.BeanUtils; import org.opennms.netmgt.config.SyslogdConfig; import org.opennms.netmgt.config.SyslogdConfigFactory; import org.opennms.netmgt.dao.db.JUnitConfigurationEnvironment; import org.opennms.netmgt.dao.db.JUnitTemporaryDatabase; import org.opennms.netmgt.mock.EventAnticipator; import org.opennms.netmgt.mock.MockDatabase; import org.opennms.netmgt.mock.MockEventIpcManager; import org.opennms.netmgt.model.events.EventBuilder; import org.opennms.netmgt.xml.event.Event; import org.springframework.beans.factory.InitializingBean; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.test.context.ContextConfiguration; import org.springframework.transaction.annotation.Transactional; @RunWith(OpenNMSJUnit4ClassRunner.class) @ContextConfiguration(locations={ "classpath:/META-INF/opennms/applicationContext-dao.xml", "classpath*:/META-INF/opennms/component-dao.xml", "classpath:/META-INF/opennms/applicationContext-daemon.xml", "classpath:/META-INF/opennms/mockEventIpcManager.xml", "classpath:/META-INF/opennms/applicationContext-setupIpLike-enabled.xml", "classpath:/META-INF/opennms/applicationContext-databasePopulator.xml", "classpath:/META-INF/opennms/applicationContext-minimal-conf.xml" }) @JUnitConfigurationEnvironment @JUnitTemporaryDatabase(dirtiesContext=false,tempDbClass=MockDatabase.class) @Transactional public class Nms4335Test implements InitializingBean { String m_localhost = "127.0.0.1"; private Syslogd m_syslogd; @Autowired private MockEventIpcManager m_eventIpcManager; private final List<ExecutorService> m_executorServices = Arrays.asList(new ExecutorService[] { Executors.newFixedThreadPool(3), Executors.newFixedThreadPool(3) }); @Override public void afterPropertiesSet() throws Exception { BeanUtils.assertAutowiring(this); } @Before public void setUp() throws Exception { MockLogAppender.setupLogging(true, "TRACE"); InputStream stream = null; try { final String config = "<?xml version=\"1.0\"?> \n" + "<syslogd-configuration> \n" + " <configuration \n" + " syslog-port=\"10514\" \n" + " new-suspect-on-message=\"false\" \n" + " forwarding-regexp=\"^((.+?) (.*))\\n?$\" \n" + " matching-group-host=\"2\" \n" + " matching-group-message=\"3\" \n" + " discard-uei=\"DISCARD-MATCHING-MESSAGES\" \n" + " /> \n" + "\n" + " <!-- Use the following to convert UEI ad-hoc --> \n" + " <ueiList> \n" + " <ueiMatch> \n" + " <match type=\"substr\" expression=\"CRISCO\"/> \n" + " <uei>CISCO</uei> \n" + " </ueiMatch> \n" + " <ueiMatch> \n" + " <match type=\"regex\" expression=\".*su:auth.*authentication failure.*\"/> \n" + " <uei>uei.opennms.org/syslog/pam/su/suFailure</uei> \n" + " </ueiMatch> \n" + " <!-- Use the following to discard a syslog message without ever creating an event for it. \n" + " If you change the value of \"discard-uei\" above, you must change the UEI used here to match. --> \n" + " <ueiMatch> \n" + " <match type=\"substr\" expression=\"JUNK\"/> \n" + " <uei>DISCARD-MATCHING-MESSAGES</uei> \n" + " </ueiMatch> \n" + " </ueiList> \n" + "\n" + " <!-- Use the following to remove a syslog message from the event-trail --> \n" + "\n" + " <hideMessage> \n" + " <hideMatch> \n" + " <match type=\"substr\" expression=\"SECRET\"/> \n" + " </hideMatch> \n" + " <hideMatch> \n" + " <match type=\"regex\" expression=\".*(double|triple)secret.*\"/> \n" + " </hideMatch> \n" + " </hideMessage> \n" + "\n" + "</syslogd-configuration>\n"; stream = new ByteArrayInputStream(config.getBytes()); SyslogdConfigFactory.setInstance(new SyslogdConfigFactory(stream)); } finally { if (stream != null) { IOUtils.closeQuietly(stream); } } m_syslogd = new Syslogd(); m_syslogd.init(); } @After public void tearDown() throws Exception { MockLogAppender.assertNotGreaterOrEqual(Level.FATAL); } @Test public void testAuthFailureShouldLog() throws Exception { doMessageTest("Jan 7 12:42:46 192.168.0.1 su[25856]: pam_unix(su:auth): authentication failure; logname=jeffg uid=1004 euid=0 tty=pts/1 ruser=jeffg rhost= user=root", "192.168.0.1", "uei.opennms.org/syslog/pam/su/suFailure", "pam_unix(su:auth): authentication failure; logname=jeffg uid=1004 euid=0 tty=pts/1 ruser=jeffg rhost= user=root"); } @Test @Ignore public void testAuthFailureShouldNotLog() throws Exception { doMessageTest("Jan 7 12:42:48 cartman su[25856]: pam_authenticate: Authentication failure", "192.168.0.1", "uei.opennms.org/blah", ""); } /** * Send a raw syslog message and expect a given event as a result * * @param testPDU The raw syslog message as it would appear on the wire (just the UDP payload) * @param expectedHost The host from which the event should be resolved as originating * @param expectedUEI The expected UEI of the resulting event * @param expectedLogMsg The expected contents of the logmsg for the resulting event * * @throws UnknownHostException * @throws InterruptedException * @throws ExecutionException */ private List<Event> doMessageTest(String testPDU, String expectedHost, String expectedUEI, String expectedLogMsg) throws UnknownHostException, InterruptedException, ExecutionException { startSyslogdGracefully(); final EventBuilder expectedEventBldr = new EventBuilder(expectedUEI, "syslogd"); expectedEventBldr.setInterface(addr(expectedHost)); expectedEventBldr.setLogDest("logndisplay"); expectedEventBldr.setLogMessage(expectedLogMsg); final EventAnticipator ea = new EventAnticipator(); m_eventIpcManager.addEventListener(ea); ea.anticipateEvent(expectedEventBldr.getEvent()); final SyslogClient sc = new SyslogClient(null, 10, SyslogClient.LOG_DAEMON); final DatagramPacket pkt = sc.getPacket(SyslogClient.LOG_DEBUG, testPDU); final SyslogdConfig config = SyslogdConfigFactory.getInstance(); WaterfallExecutor.waterfall(m_executorServices, new SyslogConnection(pkt, config.getForwardingRegexp(), config.getMatchingGroupHost(), config.getMatchingGroupMessage(), config.getUeiList(), config.getHideMessages(), config.getDiscardUei())); ea.verifyAnticipated(5000,0,0,0,0); final Event receivedEvent = ea.getAnticipatedEventsRecieved().get(0); assertEquals("Log messages do not match", expectedLogMsg, receivedEvent.getLogmsg().getContent()); return ea.getAnticipatedEventsRecieved(); } private void startSyslogdGracefully() { try { m_syslogd.start(); } catch (UndeclaredThrowableException ute) { if (ute.getCause() instanceof BindException) { // continue, this was expected } else { throw ute; } } } }