Source code for ansible_runner.loader

#
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements.  See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership.  The ASF licenses this file
# to you 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.
#

from __future__ import annotations

import os
import json
import codecs

from typing import Any, Dict
from yaml import safe_load, YAMLError

from ansible_runner.exceptions import ConfigurationError
from ansible_runner.output import debug


[docs]class ArtifactLoader: ''' Handles loading and caching file contents from disk This class will load the file contents and attempt to deserialize the contents as either JSON or YAML. If the file contents cannot be deserialized, the contents will be returned to the caller as a string. The deserialized file contents are stored as a cached object in the instance to avoid any additional reads from disk for subsequent calls to load the same file. ''' def __init__(self, base_path: str): self._cache: Dict[str, Any] = {} self.base_path = base_path def _load_json(self, contents: str) -> dict | None: ''' Attempts to deserialize the contents of a JSON object :param str contents: The contents to deserialize. :return: A dict if the contents are JSON serialized, otherwise returns None. ''' try: return json.loads(contents) except ValueError: return None def _load_yaml(self, contents: str) -> dict | None: ''' Attempts to deserialize the contents of a YAML object. :param str contents: The contents to deserialize. :return: A dict if the contents are YAML serialized, otherwise returns None. ''' try: return safe_load(contents) except YAMLError: return None def _get_contents(self, path: str) -> str: ''' Loads the contents of the file specified by path :param str path: The relative or absolute path to the file to be loaded. If the path is relative, then it is combined with the base_path to generate a full path string :return: The contents of the file as a string :raises: ConfigurationError if the file cannot be loaded. ''' try: if not os.path.exists(path): raise ConfigurationError(f"specified path does not exist {path}") with codecs.open(path, encoding='utf-8') as f: data = f.read() return data except (IOError, OSError) as exc: raise ConfigurationError(f"error trying to load file contents: {exc}") from exc
[docs] def abspath(self, path: str) -> str: ''' Transform the path to an absolute path :param str path: The path to transform to an absolute path :return: The absolute path to the file. ''' if not path.startswith(os.path.sep) or path.startswith('~'): path = os.path.expanduser(os.path.join(self.base_path, path)) return path
[docs] def isfile(self, path: str) -> bool: ''' Check if the path is a file :param str path: The path to the file to check. If the path is relative it will be exanded to an absolute path :return: True if path is a file, False otherwise. ''' return os.path.isfile(self.abspath(path))
[docs] def load_file(self, path: str, objtype: Any | None = None, encoding='utf-8') -> bytes | str | dict | None: ''' Load the file specified by path This method will first try to load the file contents from cache and if there is a cache miss, it will load the contents from disk :param str path: The full or relative path to the file to be loaded. :param Any objtype: The object type of the file contents. This is used to type check the deserialized content against the contents loaded from disk. Ignore serializing if objtype is str. Only Mapping or str types are supported. :param str encoding: The file contents text encoding. :return: The deserialized file contents which could be either a string object or a dict object :raises: ConfigurationError on error during file load or deserialization. ''' parsed_data: bytes | str | dict | None path = self.abspath(path) debug(f"file path is {path}") if path in self._cache: return self._cache[path] try: debug(f"cache miss, attempting to load file from disk: {path}") contents = parsed_data = self._get_contents(path) if encoding: parsed_data = contents.encode(encoding) except ConfigurationError as exc: debug(str(exc)) raise except UnicodeEncodeError as exc: raise ConfigurationError('unable to encode file contents') from exc if objtype is not str: for deserializer in (self._load_json, self._load_yaml): parsed_data = deserializer(contents) if parsed_data: break if objtype and not isinstance(parsed_data, objtype): debug(f"specified file {path} is not of type {objtype}") raise ConfigurationError('invalid file serialization type for contents') self._cache[path] = parsed_data return parsed_data