Skip to content

Commit

Permalink
Merge branch '6.15.z' into cherry-pick-6.15.z-e34528585ec1e991a8f5a3d…
Browse files Browse the repository at this point in the history
…3505d352c3aa0427e
  • Loading branch information
synkd authored Nov 5, 2024
2 parents 7c02e9a + df130d0 commit c985671
Show file tree
Hide file tree
Showing 17 changed files with 199 additions and 100 deletions.
4 changes: 2 additions & 2 deletions .pre-commit-config.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ repos:
- id: check-yaml
- id: debug-statements
- repo: https://github.com/astral-sh/ruff-pre-commit
rev: v0.6.9
rev: v0.7.2
hooks:
- id: ruff
args: [--fix, --exit-non-zero-on-fix]
Expand All @@ -28,6 +28,6 @@ repos:
types: [text]
require_serial: true
- repo: https://github.com/gitleaks/gitleaks
rev: v8.20.1
rev: v8.21.2
hooks:
- id: gitleaks
10 changes: 5 additions & 5 deletions conf/ipa.yaml.template
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,10 @@ IPA:
OTP_USER: otp_user
TIME_BASED_SECRET: # update the time based token secret
DISABLED_IPA_USER: disabled_user
GROUP_USERS:
- satadmin_01
- satuser_01
USERS:
USER: satuser_01
ADMIN: satadmin_01
GROUPS:
- satadmins
- satusers
USERS: satusers
ADMINS: satadmins
KEYTAB_URL:
2 changes: 1 addition & 1 deletion pytest_fixtures/component/satellite_auth.py
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,7 @@ def ipa_data():
'ldap_hostname': settings.ipa.hostname,
'time_based_secret': settings.ipa.time_based_secret,
'disabled_user_ipa': settings.ipa.disabled_ipa_user,
'group_users': settings.ipa.group_users,
'users': settings.ipa.users,
'groups': settings.ipa.groups,
}

Expand Down
74 changes: 64 additions & 10 deletions pytest_plugins/requirements/req_updater.py
Original file line number Diff line number Diff line change
@@ -1,15 +1,23 @@
from functools import cached_property
import subprocess
import sys


class UsageError(Exception):
"""The UsageError raised when the package manager is different from uv or pip"""

pass


class ReqUpdater:
# Installed package name as key and its counterpart in requirements file as value
package_deviates = {
'Betelgeuse': 'betelgeuse',
'broker': 'broker[docker]',
'broker': 'broker[docker,podman,hussh]',
'dynaconf': 'dynaconf[vault]',
'Jinja2': 'jinja2',
'Sphinx': 'sphinx',
'pyyaml': 'PyYAML',
}

@cached_property
Expand All @@ -18,9 +26,9 @@ def installed_packages(self):
This also normalizes any package names that deviates in requirements file vs installed names
"""
installed_pkges = subprocess.run(
'pip list --format=freeze', capture_output=True, shell=True
).stdout.decode()
installer_args = [sys.executable, '-m', 'list', '--format=freeze']
installer_args[2:2] = self.packagae_manager.split(' ')
installed_pkges = subprocess.run(installer_args, capture_output=True).stdout.decode()
for pkg in self.package_deviates:
if pkg in installed_pkges:
# Replacing the installed package names with their names in requirements file
Expand Down Expand Up @@ -56,16 +64,62 @@ def opt_deviation(self):
"""Returns new and updates available packages in requirements-optional file"""
return set(self.optional_packages).difference(self.installed_packages)

@cached_property
def packagae_manager(self):
if (
subprocess.run(
[sys.executable, '-m', 'pip', '--version'],
stdout=subprocess.DEVNULL,
stderr=subprocess.DEVNULL,
).returncode
== 0
):
_manager = 'pip'
elif (
subprocess.run(
[sys.executable, '-m', 'uv', '--version'],
stdout=subprocess.DEVNULL,
stderr=subprocess.DEVNULL,
).returncode
== 0
):
_manager = 'uv pip'
else:
raise UsageError(
'The package manager is not identifiable for performing package updates.'
'Currently only pip and uv is supported. Please manually update the packages '
'and rerun pytest command.'
)
return _manager

def install_req_deviations(self):
"""Installs new and updates available packages in requirements file"""
if self.req_deviation:
subprocess.run(
f"pip install {' '.join(self.req_deviation)}", shell=True, stdout=subprocess.PIPE
)
lst_of_reqs = ' '.join(f"'{req}'" for req in self.req_deviation)
if (
subprocess.run(
f"{self.packagae_manager} install {lst_of_reqs}",
shell=True,
stdout=subprocess.PIPE,
).returncode
== 0
):
print('Mandatory requirements are updated.')
else:
print('ERROR: Some issue occurred with updating the required requirements')

def install_opt_deviations(self):
"""Installs new and updates available packages in requirements-optional file"""
if self.opt_deviation:
subprocess.run(
f"pip install {' '.join(self.opt_deviation)}", shell=True, stdout=subprocess.PIPE
)
lst_of_reqs = ' '.join(f"'{req}'" for req in self.opt_deviation)
if (
subprocess.run(
f"{self.packagae_manager} install {lst_of_reqs}",
shell=True,
stdout=subprocess.PIPE,
).returncode
== 0
):
print('Optional requirements are updated.')
else:
print('ERROR: Some issue occurred with updating the optional requirements')
36 changes: 23 additions & 13 deletions pytest_plugins/requirements/update_requirements.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,12 @@
updater = ReqUpdater()


def git_deviation_filter(deviation):
"""Packages installed from Git branch and the version cant be compared, so ignore them from reporting"""
git_packages = ['nailgun', 'airgun']
return all(git_pckg not in deviation for git_pckg in git_packages)


def pytest_addoption(parser):
"""Options to allow user to update the requirements"""
update_options = {
Expand All @@ -28,24 +34,28 @@ def pytest_report_header(config):
# Following will update the mandatory and optional requirements
# pytest tests/foreman --collect-only --update-all-reqs
"""
if updater.req_deviation:
print(f"Mandatory Requirements Mismatch: {' '.join(updater.req_deviation)}")
if config.getoption('update_required_reqs') or config.getoption('update_all_reqs'):
updater.install_req_deviations()
print('Mandatory requirements are installed to be up-to-date.')
req_deviations = updater.req_deviation
filtered_req_deviations = list(filter(git_deviation_filter, req_deviations))
if filtered_req_deviations:
print(f"Mandatory Requirements Mismatch: {' '.join(filtered_req_deviations)}")
else:
print('Mandatory Requirements are up to date.')

if updater.opt_deviation:
print(f"Optional Requirements Mismatch: {' '.join(updater.opt_deviation)}")
if config.getoption('update_all_reqs'):
updater.install_opt_deviations()
print('Optional requirements are installed to be up-to-date.')
if req_deviations and (
config.getoption('update_required_reqs') or config.getoption('update_all_reqs')
):
updater.install_req_deviations()

opt_deviations = updater.opt_deviation
filtered_opt_deviations = list(filter(git_deviation_filter, opt_deviations))
if filtered_opt_deviations:
print(f"Optional Requirements Mismatch: {' '.join(filtered_opt_deviations)}")
else:
print('Optional Requirements are up to date.')
if opt_deviations and config.getoption('update_all_reqs'):
updater.install_opt_deviations()

if updater.req_deviation or updater.opt_deviation:
if req_deviations or opt_deviations:
print(
"To update mismatched requirements, run the pytest command with "
"To update requirements, run the pytest with "
"'--update-required-reqs' OR '--update-all-reqs' option."
)
2 changes: 1 addition & 1 deletion requirements-optional.txt
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ ruff==0.6.9

# For generating documentation.
sphinx==8.1.3
sphinx-autoapi==3.3.2
sphinx-autoapi==3.3.3

# For 'manage' interactive shell
manage==0.1.15
4 changes: 2 additions & 2 deletions requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -10,14 +10,14 @@ fauxfactory==3.1.1
jinja2==3.1.4
manifester==0.2.8
navmazing==1.2.2
productmd==1.40
productmd==1.41
pyotp==2.9.0
python-box==7.2.0
pytest==8.3.3
pytest-order==1.3.0
pytest-services==2.2.1
pytest-mock==3.14.0
pytest-reportportal==5.4.4
pytest-reportportal==5.4.5
pytest-xdist==3.6.1
pytest-fixturecollection==0.1.2
pytest-ibutsu==2.2.4
Expand Down
6 changes: 5 additions & 1 deletion robottelo/config/validators.py
Original file line number Diff line number Diff line change
Expand Up @@ -172,8 +172,12 @@
'ipa.user',
'ipa.otp_user',
'ipa.disabled_ipa_user',
'ipa.group_users',
'ipa.users',
'ipa.users.user',
'ipa.users.admin',
'ipa.groups',
'ipa.groups.users',
'ipa.groups.admins',
'ipa.keytab_url',
'ipa.time_based_secret',
must_exist=True,
Expand Down
2 changes: 1 addition & 1 deletion robottelo/hosts.py
Original file line number Diff line number Diff line change
Expand Up @@ -2652,7 +2652,7 @@ def __init__(self, sat_obj, **kwargs):
'base_dn': settings.ipa.basedn,
'disabled_user_ipa': settings.ipa.disabled_ipa_user,
'group_base_dn': settings.ipa.grpbasedn,
'group_users': settings.ipa.group_users,
'users': settings.ipa.users,
'groups': settings.ipa.groups,
'ipa_otp_username': settings.ipa.otp_user,
'ldap_user_cn': settings.ipa.username,
Expand Down
15 changes: 12 additions & 3 deletions robottelo/utils/datafactory.py
Original file line number Diff line number Diff line change
Expand Up @@ -477,26 +477,35 @@ def valid_http_credentials(url_encoded=False):
'quote': False,
'http_valid': True,
},
{'login': 'admin', 'pass': '', 'quote': False, 'http_valid': False},
{'login': '', 'pass': 'mypassword', 'quote': False, 'http_valid': False},
{'login': '', 'pass': '', 'quote': False, 'http_valid': False},
{'login': 'admin', 'pass': '', 'quote': False, 'http_valid': False, 'yum_compatible': True},
{
'login': '',
'pass': 'mypassword',
'quote': False,
'http_valid': False,
'yum_compatible': False,
},
{'login': '', 'pass': '', 'quote': False, 'http_valid': False, 'yum_compatible': True},
{
'login': gen_string('alpha', gen_integer(1, 512)),
'pass': gen_string('alpha'),
'quote': False,
'http_valid': False,
'yum_compatible': False,
},
{
'login': gen_string('alphanumeric', gen_integer(1, 512)),
'pass': gen_string('alphanumeric'),
'quote': False,
'http_valid': False,
'yum_compatible': False,
},
{
'login': gen_string('utf8', gen_integer(1, 50)),
'pass': gen_string('utf8'),
'quote': True,
'http_valid': False,
'yum_compatible': False,
'encoding': 'utf8',
},
]
Expand Down
9 changes: 4 additions & 5 deletions robottelo/utils/io/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,11 +18,10 @@ def get_local_file_data(path):
checksum = hashlib.sha256(file_content).hexdigest()

try:
tarobj = tarfile.open(path, mode='r')
host_counts = get_host_counts(tarobj)
tarobj.close()
extractable = True
json_files_parsable = True
with tarfile.open(path, mode='r') as tarobj:
host_counts = get_host_counts(tarobj)
extractable = True
json_files_parsable = True
except (tarfile.TarError, json.JSONDecodeError):
host_counts = {}
extractable = False
Expand Down
2 changes: 1 addition & 1 deletion tests/foreman/api/test_provisioningtemplate.py
Original file line number Diff line number Diff line change
Expand Up @@ -187,7 +187,7 @@ def test_positive_end_to_end_crud(
# clone
template_origin = template.read_json()
# remove unique keys
unique_keys = ('updated_at', 'created_at', 'id', 'name')
unique_keys = ('updated_at', 'created_at', 'id', 'name', 'cloned_from_id')
template_origin = {
key: value for key, value in template_origin.items() if key not in unique_keys
}
Expand Down
21 changes: 10 additions & 11 deletions tests/foreman/cli/test_repository.py
Original file line number Diff line number Diff line change
Expand Up @@ -754,16 +754,14 @@ def test_positive_synchronize_auth_yum_repo(self, repo, target_sat):
'repo_options',
**parametrized(
[
(
{
'content-type': 'yum',
'url': FAKE_5_YUM_REPO,
'upstream-username': creds['login'],
'upstream-password': creds['pass'],
}
for creds in valid_http_credentials()
if not creds['http_valid']
)
{
'content-type': 'yum',
'url': FAKE_5_YUM_REPO,
'upstream-username': creds['login'],
'upstream-password': creds['pass'],
}
for creds in valid_http_credentials()
if not creds['http_valid'] and creds.get('yum_compatible')
]
),
indirect=['repo_options'],
Expand All @@ -785,7 +783,8 @@ def test_negative_synchronize_auth_yum_repo(self, repo, target_sat):
response = target_sat.cli.Task.progress(
{'id': repo_sync[0]['id']}, return_raw_response=True
)
assert "Error: 401, message='Unauthorized'" in response.stderr[1].decode('utf-8')

assert "Error: 401, message='Unauthorized'" in response.stderr

@pytest.mark.tier2
@pytest.mark.upgrade
Expand Down
9 changes: 7 additions & 2 deletions tests/foreman/destructive/test_installer.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,10 +28,15 @@ def set_random_fqdn(target_sat):
shortname = gen_string('alpha')
new_domain = gen_domain()
target_sat.execute(
f'echo "search {new_domain}" >> /etc/resolv.conf; hostnamectl set-hostname {shortname}'
'mv -f /etc/resolv.conf /etc/resolv.conf.bak; '
f'echo "search {new_domain}" > /etc/resolv.conf; '
f'hostnamectl set-hostname {shortname}'
)
yield shortname, new_domain
target_sat.execute(f'hostnamectl set-hostname {target_sat.hostname}')
target_sat.execute(
'mv -f /etc/resolv.conf.bak /etc/resolv.conf; '
f'hostnamectl set-hostname {target_sat.hostname}'
)


def test_installer_sat_pub_directory_accessibility(target_sat):
Expand Down
Loading

0 comments on commit c985671

Please sign in to comment.