Merge "Restoring logging facility and implementing default logger"
This commit is contained in:
commit
0dcf77de8c
@ -16,7 +16,9 @@
|
|||||||
|
|
||||||
import argparse
|
import argparse
|
||||||
|
|
||||||
from validations_libs.utils import LOG
|
from validations_libs import utils
|
||||||
|
|
||||||
|
LOG = utils.getLogger(__name__ + '.parseractions')
|
||||||
|
|
||||||
|
|
||||||
class CommaListAction(argparse.Action):
|
class CommaListAction(argparse.Action):
|
||||||
|
47
validations_libs/logger.py
Normal file
47
validations_libs/logger.py
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
# Copyright 2022 Red Hat, Inc.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
from logging.handlers import SysLogHandler
|
||||||
|
|
||||||
|
|
||||||
|
def getLogger(loggerName, stream_lvl=logging.WARN):
|
||||||
|
"""Create logger instance.
|
||||||
|
|
||||||
|
:param loggerName: name of the new Logger instance
|
||||||
|
:type loggerName: `str`
|
||||||
|
:param stream_lvl: minimum level at which the messages will be printed to stream
|
||||||
|
:type stream_lvl: `int`
|
||||||
|
:rtype: `Logger`
|
||||||
|
"""
|
||||||
|
new_logger = logging.getLogger(loggerName)
|
||||||
|
|
||||||
|
formatter = logging.Formatter("%(asctime)s %(module)s %(message)s")
|
||||||
|
|
||||||
|
s_handler = logging.StreamHandler()
|
||||||
|
s_handler.setFormatter(formatter)
|
||||||
|
s_handler.setLevel(stream_lvl)
|
||||||
|
|
||||||
|
new_logger.addHandler(s_handler)
|
||||||
|
|
||||||
|
if os.path.exists('/dev/log'):
|
||||||
|
sys_handler = SysLogHandler(address='/dev/log')
|
||||||
|
sys_handler.setFormatter(formatter)
|
||||||
|
|
||||||
|
new_logger.addHandler(sys_handler)
|
||||||
|
else:
|
||||||
|
new_logger.warning("Journal socket does not exist. Logs will not be processed by syslog.")
|
||||||
|
|
||||||
|
return new_logger
|
44
validations_libs/tests/test_logger.py
Normal file
44
validations_libs/tests/test_logger.py
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
# Copyright 2022 Red Hat, Inc.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
||||||
|
|
||||||
|
try:
|
||||||
|
from unittest import mock
|
||||||
|
except ImportError:
|
||||||
|
import mock
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from validations_libs import logger
|
||||||
|
|
||||||
|
|
||||||
|
class TestLogger(TestCase):
|
||||||
|
|
||||||
|
def setUp(self) -> None:
|
||||||
|
super().setUp()
|
||||||
|
|
||||||
|
@mock.patch('os.path.exists', reutrn_value=True)
|
||||||
|
def test_logger_init(self, mock_exists):
|
||||||
|
new_logger = logger.getLogger("fooo")
|
||||||
|
mock_exists.assert_called_once_with('/dev/log')
|
||||||
|
self.assertEqual(logging.Logger, type(new_logger))
|
||||||
|
|
||||||
|
@mock.patch('logging.Logger.warning')
|
||||||
|
@mock.patch('os.path.exists', return_value=False)
|
||||||
|
def test_logger_init_no_journal(self, mock_exists, mock_warning):
|
||||||
|
new_logger = logger.getLogger("fooo")
|
||||||
|
mock_exists.assert_called_once_with('/dev/log')
|
||||||
|
mock_warning.assert_called_once()
|
||||||
|
self.assertEqual(logging.Logger, type(new_logger))
|
@ -40,6 +40,7 @@ class TestUtils(TestCase):
|
|||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(TestUtils, self).setUp()
|
super(TestUtils, self).setUp()
|
||||||
|
self.logger = mock.patch('validations_libs.logger.getLogger')
|
||||||
|
|
||||||
@mock.patch('validations_libs.validation.Validation._get_content',
|
@mock.patch('validations_libs.validation.Validation._get_content',
|
||||||
return_value=fakes.FAKE_PLAYBOOK[0])
|
return_value=fakes.FAKE_PLAYBOOK[0])
|
||||||
@ -399,19 +400,16 @@ class TestUtils(TestCase):
|
|||||||
[], [])
|
[], [])
|
||||||
self.assertEqual(result, {})
|
self.assertEqual(result, {})
|
||||||
|
|
||||||
@mock.patch('validations_libs.utils.LOG', autospec=True)
|
|
||||||
@mock.patch('validations_libs.utils.os.makedirs')
|
@mock.patch('validations_libs.utils.os.makedirs')
|
||||||
@mock.patch(
|
@mock.patch(
|
||||||
'validations_libs.utils.os.access',
|
'validations_libs.utils.os.access',
|
||||||
side_effect=[False, True])
|
side_effect=[False, True])
|
||||||
@mock.patch('validations_libs.utils.os.path.exists', return_value=True)
|
@mock.patch('validations_libs.utils.os.path.exists', return_value=True)
|
||||||
def test_create_log_dir_access_issue(self, mock_exists,
|
def test_create_log_dir_access_issue(self, mock_exists,
|
||||||
mock_access, mock_mkdirs,
|
mock_access, mock_mkdirs):
|
||||||
mock_log):
|
|
||||||
log_path = utils.create_log_dir("/foo/bar")
|
log_path = utils.create_log_dir("/foo/bar")
|
||||||
self.assertEqual(log_path, constants.VALIDATIONS_LOG_BASEDIR)
|
self.assertEqual(log_path, constants.VALIDATIONS_LOG_BASEDIR)
|
||||||
|
|
||||||
@mock.patch('validations_libs.utils.LOG', autospec=True)
|
|
||||||
@mock.patch(
|
@mock.patch(
|
||||||
'validations_libs.utils.os.makedirs',
|
'validations_libs.utils.os.makedirs',
|
||||||
side_effect=PermissionError)
|
side_effect=PermissionError)
|
||||||
@ -424,8 +422,7 @@ class TestUtils(TestCase):
|
|||||||
autospec=True,
|
autospec=True,
|
||||||
side_effect=fakes._accept_default_log_path)
|
side_effect=fakes._accept_default_log_path)
|
||||||
def test_create_log_dir_existence_issue(self, mock_exists,
|
def test_create_log_dir_existence_issue(self, mock_exists,
|
||||||
mock_access, mock_mkdirs,
|
mock_access, mock_mkdirs):
|
||||||
mock_log):
|
|
||||||
"""Tests behavior after encountering non-existence
|
"""Tests behavior after encountering non-existence
|
||||||
of the the selected log folder, failed attempt to create it
|
of the the selected log folder, failed attempt to create it
|
||||||
(raising PermissionError), and finally resorting to a fallback.
|
(raising PermissionError), and finally resorting to a fallback.
|
||||||
@ -433,32 +430,27 @@ class TestUtils(TestCase):
|
|||||||
log_path = utils.create_log_dir("/foo/bar")
|
log_path = utils.create_log_dir("/foo/bar")
|
||||||
self.assertEqual(log_path, constants.VALIDATIONS_LOG_BASEDIR)
|
self.assertEqual(log_path, constants.VALIDATIONS_LOG_BASEDIR)
|
||||||
|
|
||||||
@mock.patch('validations_libs.utils.LOG', autospec=True)
|
|
||||||
@mock.patch('validations_libs.utils.os.makedirs')
|
@mock.patch('validations_libs.utils.os.makedirs')
|
||||||
@mock.patch('validations_libs.utils.os.access', return_value=True)
|
@mock.patch('validations_libs.utils.os.access', return_value=True)
|
||||||
@mock.patch('validations_libs.utils.os.path.exists', return_value=True)
|
@mock.patch('validations_libs.utils.os.path.exists', return_value=True)
|
||||||
def test_create_log_dir_success(self, mock_exists,
|
def test_create_log_dir_success(self, mock_exists,
|
||||||
mock_access, mock_mkdirs,
|
mock_access, mock_mkdirs):
|
||||||
mock_log):
|
|
||||||
"""Test successful log dir retrieval on the first try.
|
"""Test successful log dir retrieval on the first try.
|
||||||
"""
|
"""
|
||||||
log_path = utils.create_log_dir("/foo/bar")
|
log_path = utils.create_log_dir("/foo/bar")
|
||||||
self.assertEqual(log_path, "/foo/bar")
|
self.assertEqual(log_path, "/foo/bar")
|
||||||
|
|
||||||
@mock.patch('validations_libs.utils.LOG', autospec=True)
|
|
||||||
@mock.patch(
|
@mock.patch(
|
||||||
'validations_libs.utils.os.makedirs',
|
'validations_libs.utils.os.makedirs',
|
||||||
side_effect=PermissionError)
|
side_effect=PermissionError)
|
||||||
@mock.patch('validations_libs.utils.os.access', return_value=False)
|
@mock.patch('validations_libs.utils.os.access', return_value=False)
|
||||||
@mock.patch('validations_libs.utils.os.path.exists', return_value=False)
|
@mock.patch('validations_libs.utils.os.path.exists', return_value=False)
|
||||||
def test_create_log_dir_runtime_err(self, mock_exists,
|
def test_create_log_dir_runtime_err(self, mock_exists,
|
||||||
mock_access, mock_mkdirs,
|
mock_access, mock_mkdirs):
|
||||||
mock_log):
|
|
||||||
"""Test if failure of the fallback raises 'RuntimeError'
|
"""Test if failure of the fallback raises 'RuntimeError'
|
||||||
"""
|
"""
|
||||||
self.assertRaises(RuntimeError, utils.create_log_dir, "/foo/bar")
|
self.assertRaises(RuntimeError, utils.create_log_dir, "/foo/bar")
|
||||||
|
|
||||||
@mock.patch('validations_libs.utils.LOG', autospec=True)
|
|
||||||
@mock.patch(
|
@mock.patch(
|
||||||
'validations_libs.utils.os.makedirs',
|
'validations_libs.utils.os.makedirs',
|
||||||
side_effect=PermissionError)
|
side_effect=PermissionError)
|
||||||
@ -468,19 +460,16 @@ class TestUtils(TestCase):
|
|||||||
side_effect=fakes._accept_default_log_path)
|
side_effect=fakes._accept_default_log_path)
|
||||||
def test_create_log_dir_default_perms_runtime_err(
|
def test_create_log_dir_default_perms_runtime_err(
|
||||||
self, mock_exists,
|
self, mock_exists,
|
||||||
mock_access, mock_mkdirs,
|
mock_access, mock_mkdirs):
|
||||||
mock_log):
|
|
||||||
"""Test if the inaccessible fallback raises 'RuntimeError'
|
"""Test if the inaccessible fallback raises 'RuntimeError'
|
||||||
"""
|
"""
|
||||||
self.assertRaises(RuntimeError, utils.create_log_dir, "/foo/bar")
|
self.assertRaises(RuntimeError, utils.create_log_dir, "/foo/bar")
|
||||||
|
|
||||||
@mock.patch('validations_libs.utils.LOG', autospec=True)
|
|
||||||
@mock.patch('validations_libs.utils.os.makedirs')
|
@mock.patch('validations_libs.utils.os.makedirs')
|
||||||
@mock.patch('validations_libs.utils.os.access', return_value=False)
|
@mock.patch('validations_libs.utils.os.access', return_value=False)
|
||||||
@mock.patch('validations_libs.utils.os.path.exists', return_value=False)
|
@mock.patch('validations_libs.utils.os.path.exists', return_value=False)
|
||||||
def test_create_log_dir_mkdirs(self, mock_exists,
|
def test_create_log_dir_mkdirs(self, mock_exists,
|
||||||
mock_access, mock_mkdirs,
|
mock_access, mock_mkdirs):
|
||||||
mock_log):
|
|
||||||
"""Test successful creation of the directory if the first access fails.
|
"""Test successful creation of the directory if the first access fails.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@ -532,7 +521,6 @@ class TestUtils(TestCase):
|
|||||||
results['ansible_environment']['ANSIBLE_STDOUT_CALLBACK'],
|
results['ansible_environment']['ANSIBLE_STDOUT_CALLBACK'],
|
||||||
fakes.ANSIBLE_ENVIRONNMENT_CONFIG['ANSIBLE_STDOUT_CALLBACK'])
|
fakes.ANSIBLE_ENVIRONNMENT_CONFIG['ANSIBLE_STDOUT_CALLBACK'])
|
||||||
|
|
||||||
@mock.patch('validations_libs.utils.LOG', autospec=True)
|
|
||||||
@mock.patch('{}.Path.exists'.format(PATHLIB),
|
@mock.patch('{}.Path.exists'.format(PATHLIB),
|
||||||
return_value=False)
|
return_value=False)
|
||||||
@mock.patch('{}.Path.is_dir'.format(PATHLIB),
|
@mock.patch('{}.Path.is_dir'.format(PATHLIB),
|
||||||
@ -543,8 +531,7 @@ class TestUtils(TestCase):
|
|||||||
def test_check_creation_community_validations_dir(self, mock_mkdir,
|
def test_check_creation_community_validations_dir(self, mock_mkdir,
|
||||||
mock_iterdir,
|
mock_iterdir,
|
||||||
mock_isdir,
|
mock_isdir,
|
||||||
mock_exists,
|
mock_exists):
|
||||||
mock_log):
|
|
||||||
basedir = PosixPath('/foo/bar/community-validations')
|
basedir = PosixPath('/foo/bar/community-validations')
|
||||||
subdir = fakes.COVAL_SUBDIR
|
subdir = fakes.COVAL_SUBDIR
|
||||||
result = utils.check_community_validations_dir(basedir, subdir)
|
result = utils.check_community_validations_dir(basedir, subdir)
|
||||||
@ -556,7 +543,6 @@ class TestUtils(TestCase):
|
|||||||
PosixPath("/foo/bar/community-validations/lookup_plugins")]
|
PosixPath("/foo/bar/community-validations/lookup_plugins")]
|
||||||
)
|
)
|
||||||
|
|
||||||
@mock.patch('validations_libs.utils.LOG', autospec=True)
|
|
||||||
@mock.patch('{}.Path.is_dir'.format(PATHLIB), return_value=True)
|
@mock.patch('{}.Path.is_dir'.format(PATHLIB), return_value=True)
|
||||||
@mock.patch('{}.Path.exists'.format(PATHLIB), return_value=True)
|
@mock.patch('{}.Path.exists'.format(PATHLIB), return_value=True)
|
||||||
@mock.patch('{}.Path.iterdir'.format(PATHLIB),
|
@mock.patch('{}.Path.iterdir'.format(PATHLIB),
|
||||||
@ -566,8 +552,7 @@ class TestUtils(TestCase):
|
|||||||
mock_mkdir,
|
mock_mkdir,
|
||||||
mock_iterdir,
|
mock_iterdir,
|
||||||
mock_exists,
|
mock_exists,
|
||||||
mock_isdir,
|
mock_isdir):
|
||||||
mock_log):
|
|
||||||
basedir = PosixPath('/foo/bar/community-validations')
|
basedir = PosixPath('/foo/bar/community-validations')
|
||||||
subdir = fakes.COVAL_SUBDIR
|
subdir = fakes.COVAL_SUBDIR
|
||||||
result = utils.check_community_validations_dir(basedir, subdir)
|
result = utils.check_community_validations_dir(basedir, subdir)
|
||||||
|
@ -16,7 +16,6 @@ import ast
|
|||||||
import configparser
|
import configparser
|
||||||
import datetime
|
import datetime
|
||||||
import glob
|
import glob
|
||||||
import logging
|
|
||||||
import os
|
import os
|
||||||
import site
|
import site
|
||||||
import subprocess
|
import subprocess
|
||||||
@ -31,8 +30,9 @@ except ImportError:
|
|||||||
from validations_libs import constants
|
from validations_libs import constants
|
||||||
from validations_libs.group import Group
|
from validations_libs.group import Group
|
||||||
from validations_libs.validation import Validation
|
from validations_libs.validation import Validation
|
||||||
|
from validations_libs.logger import getLogger
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__ + ".utils")
|
LOG = getLogger(__name__ + ".utils")
|
||||||
|
|
||||||
|
|
||||||
def current_time():
|
def current_time():
|
||||||
|
Loading…
Reference in New Issue
Block a user