2007-01-21 22:19:57 +00:00
|
|
|
#!/usr/bin/python
|
2011-10-07 19:47:50 +00:00
|
|
|
# emacs: -*- mode: python; py-indent-offset: 4; indent-tabs-mode: t -*-
|
|
|
|
# vi: set ft=python sts=4 ts=4 sw=4 noet :
|
2012-06-16 02:21:16 +00:00
|
|
|
"""Script to run Fail2Ban tests battery
|
|
|
|
"""
|
2011-10-07 19:47:50 +00:00
|
|
|
|
2006-06-26 20:05:00 +00:00
|
|
|
# This file is part of Fail2Ban.
|
|
|
|
#
|
|
|
|
# Fail2Ban 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.
|
|
|
|
#
|
|
|
|
# Fail2Ban 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 Fail2Ban; if not, write to the Free Software
|
2011-11-21 12:20:20 +00:00
|
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2006-06-26 20:05:00 +00:00
|
|
|
|
|
|
|
# Author: Cyril Jaquier
|
|
|
|
|
|
|
|
__author__ = "Cyril Jaquier"
|
2006-07-16 22:21:58 +00:00
|
|
|
__version__ = "$Revision$"
|
|
|
|
__date__ = "$Date$"
|
2006-06-26 20:05:00 +00:00
|
|
|
__copyright__ = "Copyright (c) 2004 Cyril Jaquier"
|
|
|
|
__license__ = "GPL"
|
|
|
|
|
|
|
|
|
2011-10-07 19:14:13 +00:00
|
|
|
import unittest, logging, sys, time, os
|
2006-06-26 20:05:00 +00:00
|
|
|
|
2006-11-16 21:07:42 +00:00
|
|
|
from common.version import version
|
2006-08-06 21:56:01 +00:00
|
|
|
from testcases import banmanagertestcase
|
|
|
|
from testcases import clientreadertestcase
|
|
|
|
from testcases import failmanagertestcase
|
2006-06-26 20:05:00 +00:00
|
|
|
from testcases import filtertestcase
|
|
|
|
from testcases import servertestcase
|
2006-09-05 21:16:28 +00:00
|
|
|
from testcases import datedetectortestcase
|
2006-10-23 20:13:21 +00:00
|
|
|
from testcases import actiontestcase
|
2006-10-18 22:35:32 +00:00
|
|
|
from server.mytime import MyTime
|
|
|
|
|
2012-06-16 02:21:16 +00:00
|
|
|
from optparse import OptionParser, Option
|
|
|
|
|
|
|
|
def get_opt_parser():
|
|
|
|
# use module docstring for help output
|
|
|
|
p = OptionParser(
|
|
|
|
usage="%s [OPTIONS]\n" % sys.argv[0] + __doc__,
|
|
|
|
version="%prog " + version)
|
|
|
|
|
|
|
|
p.add_options([
|
|
|
|
Option('-l', "--log-level", type="choice",
|
|
|
|
dest="log_level",
|
2012-06-30 04:35:43 +00:00
|
|
|
choices=('debug', 'info', 'warn', 'error', 'fatal'),
|
|
|
|
default=None,
|
2012-06-16 02:21:16 +00:00
|
|
|
help="Log level for the logger to use during running tests"),
|
|
|
|
])
|
|
|
|
|
|
|
|
return p
|
|
|
|
|
|
|
|
parser = get_opt_parser()
|
|
|
|
(opts, files) = parser.parse_args()
|
|
|
|
assert(not len(files))
|
|
|
|
|
2012-07-19 17:30:55 +00:00
|
|
|
#
|
|
|
|
# Logging
|
|
|
|
#
|
|
|
|
logSys = logging.getLogger("fail2ban")
|
2011-10-07 19:14:13 +00:00
|
|
|
|
2012-07-19 17:30:55 +00:00
|
|
|
# Numerical level of verbosity corresponding to a log "level"
|
|
|
|
verbosity = {'debug': 3,
|
|
|
|
'info': 2,
|
|
|
|
'warn': 1,
|
|
|
|
'error': 1,
|
|
|
|
'fatal': 0,
|
|
|
|
None: 1}[opts.log_level]
|
2006-06-26 20:05:00 +00:00
|
|
|
|
2012-06-30 04:35:43 +00:00
|
|
|
if opts.log_level is not None:
|
|
|
|
# so we had explicit settings
|
|
|
|
logSys.setLevel(getattr(logging, opts.log_level.upper()))
|
|
|
|
else:
|
|
|
|
# suppress the logging but it would leave unittests' progress dots
|
|
|
|
# ticking, unless like with '-l fatal' which would be silent
|
|
|
|
# unless error occurs
|
|
|
|
logSys.setLevel(getattr(logging, 'FATAL'))
|
|
|
|
|
2012-07-19 17:30:55 +00:00
|
|
|
# Add the default logging handler
|
|
|
|
stdout = logging.StreamHandler(sys.stdout)
|
|
|
|
# Custom log format for the verbose tests runs
|
|
|
|
if verbosity > 1:
|
|
|
|
stdout.setFormatter(logging.Formatter(' %(asctime)-15s %(thread)s %(message)s'))
|
|
|
|
else:
|
|
|
|
# just prefix with the space
|
|
|
|
stdout.setFormatter(logging.Formatter(' %(message)s'))
|
|
|
|
logSys.addHandler(stdout)
|
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Let know the version
|
|
|
|
#
|
2012-06-30 04:35:43 +00:00
|
|
|
if not opts.log_level or opts.log_level != 'fatal':
|
|
|
|
print "Fail2ban " + version + " test suite. Please wait..."
|
2006-06-26 20:05:00 +00:00
|
|
|
|
2012-07-19 17:30:55 +00:00
|
|
|
|
|
|
|
#
|
|
|
|
# Gather the tests
|
|
|
|
#
|
2006-06-26 20:05:00 +00:00
|
|
|
tests = unittest.TestSuite()
|
|
|
|
|
|
|
|
# Server
|
|
|
|
#tests.addTest(unittest.makeSuite(servertestcase.StartStop))
|
|
|
|
#tests.addTest(unittest.makeSuite(servertestcase.Transmitter))
|
2006-10-23 20:13:21 +00:00
|
|
|
tests.addTest(unittest.makeSuite(actiontestcase.ExecuteAction))
|
2006-06-26 20:05:00 +00:00
|
|
|
# FailManager
|
|
|
|
tests.addTest(unittest.makeSuite(failmanagertestcase.AddFailure))
|
|
|
|
# BanManager
|
|
|
|
tests.addTest(unittest.makeSuite(banmanagertestcase.AddFailure))
|
2006-08-06 21:56:01 +00:00
|
|
|
# ClientReader
|
|
|
|
tests.addTest(unittest.makeSuite(clientreadertestcase.JailReaderTest))
|
2012-06-29 16:56:32 +00:00
|
|
|
|
2012-07-20 03:08:33 +00:00
|
|
|
# Filter
|
|
|
|
tests.addTest(unittest.makeSuite(filtertestcase.IgnoreIP))
|
|
|
|
tests.addTest(unittest.makeSuite(filtertestcase.LogFile))
|
|
|
|
tests.addTest(unittest.makeSuite(filtertestcase.LogFileMonitor))
|
|
|
|
tests.addTest(unittest.makeSuite(filtertestcase.GetFailures))
|
|
|
|
tests.addTest(unittest.makeSuite(filtertestcase.DNSUtilsTests))
|
|
|
|
|
|
|
|
# DateDetector
|
|
|
|
tests.addTest(unittest.makeSuite(datedetectortestcase.DateDetectorTest))
|
|
|
|
|
|
|
|
#
|
|
|
|
# Extensive use-tests of different available filters backends
|
|
|
|
#
|
|
|
|
|
2012-06-29 16:56:32 +00:00
|
|
|
from server.filterpoll import FilterPoll
|
|
|
|
filters = [FilterPoll] # always available
|
|
|
|
|
|
|
|
# Additional filters available only if external modules are available
|
|
|
|
# yoh: Since I do not know better way for parametric tests
|
|
|
|
# with good old unittest
|
|
|
|
try:
|
|
|
|
from server.filtergamin import FilterGamin
|
2012-07-20 03:08:33 +00:00
|
|
|
filters.append(FilterGamin)
|
2012-06-29 16:56:32 +00:00
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
try:
|
|
|
|
from server.filterpyinotify import FilterPyinotify
|
|
|
|
filters.append(FilterPyinotify)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
for Filter_ in filters:
|
|
|
|
tests.addTest(unittest.makeSuite(
|
|
|
|
filtertestcase.get_monitor_failures_testcase(Filter_)))
|
|
|
|
|
2012-07-19 17:30:55 +00:00
|
|
|
|
|
|
|
#
|
|
|
|
# Run the tests
|
|
|
|
#
|
|
|
|
testRunner = unittest.TextTestRunner(verbosity=verbosity)
|
|
|
|
|
|
|
|
try:
|
|
|
|
# Set the time to a fixed, known value
|
|
|
|
# Sun Aug 14 12:00:00 CEST 2005
|
|
|
|
# yoh: we need to adjust TZ to match the one used by Cyril so all the timestamps match
|
|
|
|
old_TZ = os.environ.get('TZ', None)
|
|
|
|
os.environ['TZ'] = 'Europe/Zurich'
|
|
|
|
time.tzset()
|
|
|
|
MyTime.setTime(1124013600)
|
|
|
|
|
|
|
|
tests_results = testRunner.run(tests)
|
|
|
|
|
|
|
|
finally:
|
|
|
|
# Just for the sake of it reset the TZ
|
|
|
|
# yoh: move all this into setup/teardown methods within tests
|
|
|
|
os.environ.pop('TZ')
|
|
|
|
if old_TZ:
|
|
|
|
os.environ['TZ'] = old_TZ
|
|
|
|
time.tzset()
|
|
|
|
|
|
|
|
if not tests_results.wasSuccessful():
|
|
|
|
sys.exit(1)
|