Module rok4_tools.pyr2pyr

Expand source code
#!/usr/bin/env python3

import argparse
import json
import logging
import os
import sys
from json.decoder import JSONDecodeError

from jsonschema import ValidationError, validate
from rok4 import storage

from rok4_tools import __version__
from rok4_tools.pyr2pyr_utils.agent import work as agent_work
from rok4_tools.pyr2pyr_utils.finisher import work as finisher_work
from rok4_tools.pyr2pyr_utils.master import work as master_work

# Default logger
logging.basicConfig(format="%(asctime)s %(levelname)s: %(message)s", level=logging.WARNING)

config = {}
args = None


def parse() -> None:
    """Parse call arguments and check values

    Exit program if an error occured
    """

    global args

    # CLI call parser
    parser = argparse.ArgumentParser(
        prog="pyr2pyr",
        description="Tool to split the work to do for a pyramid copy and make slabs' copy",
        epilog="",
    )

    parser.add_argument("--version", action="version", version="%(prog)s " + __version__)

    parser.add_argument(
        "--role",
        choices=["master", "agent", "finisher", "example", "check"],
        action="store",
        dest="role",
        help="Script's role",
        required=True,
    )

    parser.add_argument(
        "--conf",
        metavar="storage://path/to/conf.json",
        action="store",
        dest="configuration",
        help="Configuration file or object, JSON format",
        required=False,
    )

    parser.add_argument(
        "--split",
        type=int,
        metavar="N",
        action="store",
        dest="split",
        help="Split number, only required for the agent role",
        required=False,
    )

    args = parser.parse_args()

    if args.role != "example" and (args.configuration is None):
        print("pyr2pyr: error: argument --conf is required for all roles except 'example'")
        sys.exit(1)

    if args.role == "agent" and (args.split is None or args.split < 1):
        print(
            "pyr2pyr: error: argument --split is required for the agent role and have to be a positive integer"
        )
        sys.exit(1)


def configuration() -> None:
    """Load configuration file

    Raises:
        JSONDecodeError: Configuration is not a valid JSON file
        ValidationError: Configuration is not a valid PYR2PYR configuration file
        MissingEnvironmentError: Missing object storage informations
        StorageError: storage read issue
        FileNotFoundError: File or object does not exist
    """

    global config

    # Chargement du schéma JSON
    f = open(os.path.join(os.path.dirname(__file__), "pyr2pyr_utils", "schema.json"))
    schema = json.load(f)
    f.close()

    # Chargement et validation de la configuration JSON
    config = json.loads(storage.get_data_str(args.configuration))
    validate(config, schema)

    # Valeurs par défaut et cohérence avec l'appel
    if config["to"]["storage"]["type"] == "FILE" and "depth" not in config["to"]["storage"]:
        config["to"]["storage"]["depth"] = 2

    if "parallelization" not in config["process"]:
        config["process"]["parallelization"] = 1

    if args.role == "agent" and args.split > config["process"]["parallelization"]:
        raise Exception(
            f"Split number have to be consistent with the parallelization level: {args.split} > {config['process']['parallelization']}"
        )

    if "follow_links" not in config["process"]:
        config["process"]["follow_links"] = False

    if "slab_limit" not in config["process"]:
        config["process"]["slab_limit"] = 0

    # Logger
    if "logger" in config:
        # On supprime l'ancien logger (celui configuré par défaut) et on le reconfigure avec les nouveaux paramètres
        for handler in logging.root.handlers[:]:
            logging.root.removeHandler(handler)

        if "file" in config["logger"]:
            logging.basicConfig(
                level=logging.getLevelName(config["logger"].get("level", "WARNING")),
                format=config["logger"].get("layout", "%(asctime)s %(levelname)s: %(message)s"),
                filename=config["logger"]["file"],
            )
        else:
            logging.basicConfig(
                level=logging.getLevelName(config["logger"].get("level", "WARNING")),
                format=config["logger"].get("layout", "%(asctime)s %(levelname)s: %(message)s"),
            )


def main() -> None:
    """Main function

    Return 0 if success, 1 if an error occured
    """

    parse()

    if args.role == "example":
        # On veut juste afficher la configuration en exemple
        f = open(os.path.join(os.path.dirname(__file__), "pyr2pyr_utils/example.json"))
        print(f.read())
        f.close
        sys.exit(0)

    # Configuration
    try:
        configuration()

    except JSONDecodeError as e:
        logging.error(f"{args.configuration} is not a valid JSON file: {e}")
        sys.exit(1)

    except ValidationError as e:
        logging.error(f"{args.configuration} is not a valid configuration file: {e}")
        sys.exit(1)

    except FileNotFoundError as e:
        logging.error(f"{e} does not exists")
        sys.exit(1)

    except Exception as e:
        logging.error(e)
        sys.exit(1)

    if args.role == "check":
        # On voulait juste valider le fichier de configuration, c'est chose faite
        # Si on est là c'est que tout est bon
        print("Valid configuration !")
        sys.exit(0)

    # Work
    try:
        if args.role == "master":
            master_work(config)
        elif args.role == "agent":
            agent_work(config, args.split)
        elif args.role == "finisher":
            finisher_work(config)

    except Exception as e:
        logging.error(e)
        sys.exit(1)

    sys.exit(0)


if __name__ == "__main__":
    main()

Functions

def configuration() ‑> None

Load configuration file

Raises

JSONDecodeError
Configuration is not a valid JSON file
ValidationError
Configuration is not a valid PYR2PYR configuration file
MissingEnvironmentError
Missing object storage informations
StorageError
storage read issue
FileNotFoundError
File or object does not exist
Expand source code
def configuration() -> None:
    """Load configuration file

    Raises:
        JSONDecodeError: Configuration is not a valid JSON file
        ValidationError: Configuration is not a valid PYR2PYR configuration file
        MissingEnvironmentError: Missing object storage informations
        StorageError: storage read issue
        FileNotFoundError: File or object does not exist
    """

    global config

    # Chargement du schéma JSON
    f = open(os.path.join(os.path.dirname(__file__), "pyr2pyr_utils", "schema.json"))
    schema = json.load(f)
    f.close()

    # Chargement et validation de la configuration JSON
    config = json.loads(storage.get_data_str(args.configuration))
    validate(config, schema)

    # Valeurs par défaut et cohérence avec l'appel
    if config["to"]["storage"]["type"] == "FILE" and "depth" not in config["to"]["storage"]:
        config["to"]["storage"]["depth"] = 2

    if "parallelization" not in config["process"]:
        config["process"]["parallelization"] = 1

    if args.role == "agent" and args.split > config["process"]["parallelization"]:
        raise Exception(
            f"Split number have to be consistent with the parallelization level: {args.split} > {config['process']['parallelization']}"
        )

    if "follow_links" not in config["process"]:
        config["process"]["follow_links"] = False

    if "slab_limit" not in config["process"]:
        config["process"]["slab_limit"] = 0

    # Logger
    if "logger" in config:
        # On supprime l'ancien logger (celui configuré par défaut) et on le reconfigure avec les nouveaux paramètres
        for handler in logging.root.handlers[:]:
            logging.root.removeHandler(handler)

        if "file" in config["logger"]:
            logging.basicConfig(
                level=logging.getLevelName(config["logger"].get("level", "WARNING")),
                format=config["logger"].get("layout", "%(asctime)s %(levelname)s: %(message)s"),
                filename=config["logger"]["file"],
            )
        else:
            logging.basicConfig(
                level=logging.getLevelName(config["logger"].get("level", "WARNING")),
                format=config["logger"].get("layout", "%(asctime)s %(levelname)s: %(message)s"),
            )
def main() ‑> None

Main function

Return 0 if success, 1 if an error occured

Expand source code
def main() -> None:
    """Main function

    Return 0 if success, 1 if an error occured
    """

    parse()

    if args.role == "example":
        # On veut juste afficher la configuration en exemple
        f = open(os.path.join(os.path.dirname(__file__), "pyr2pyr_utils/example.json"))
        print(f.read())
        f.close
        sys.exit(0)

    # Configuration
    try:
        configuration()

    except JSONDecodeError as e:
        logging.error(f"{args.configuration} is not a valid JSON file: {e}")
        sys.exit(1)

    except ValidationError as e:
        logging.error(f"{args.configuration} is not a valid configuration file: {e}")
        sys.exit(1)

    except FileNotFoundError as e:
        logging.error(f"{e} does not exists")
        sys.exit(1)

    except Exception as e:
        logging.error(e)
        sys.exit(1)

    if args.role == "check":
        # On voulait juste valider le fichier de configuration, c'est chose faite
        # Si on est là c'est que tout est bon
        print("Valid configuration !")
        sys.exit(0)

    # Work
    try:
        if args.role == "master":
            master_work(config)
        elif args.role == "agent":
            agent_work(config, args.split)
        elif args.role == "finisher":
            finisher_work(config)

    except Exception as e:
        logging.error(e)
        sys.exit(1)

    sys.exit(0)
def parse() ‑> None

Parse call arguments and check values

Exit program if an error occured

Expand source code
def parse() -> None:
    """Parse call arguments and check values

    Exit program if an error occured
    """

    global args

    # CLI call parser
    parser = argparse.ArgumentParser(
        prog="pyr2pyr",
        description="Tool to split the work to do for a pyramid copy and make slabs' copy",
        epilog="",
    )

    parser.add_argument("--version", action="version", version="%(prog)s " + __version__)

    parser.add_argument(
        "--role",
        choices=["master", "agent", "finisher", "example", "check"],
        action="store",
        dest="role",
        help="Script's role",
        required=True,
    )

    parser.add_argument(
        "--conf",
        metavar="storage://path/to/conf.json",
        action="store",
        dest="configuration",
        help="Configuration file or object, JSON format",
        required=False,
    )

    parser.add_argument(
        "--split",
        type=int,
        metavar="N",
        action="store",
        dest="split",
        help="Split number, only required for the agent role",
        required=False,
    )

    args = parser.parse_args()

    if args.role != "example" and (args.configuration is None):
        print("pyr2pyr: error: argument --conf is required for all roles except 'example'")
        sys.exit(1)

    if args.role == "agent" and (args.split is None or args.split < 1):
        print(
            "pyr2pyr: error: argument --split is required for the agent role and have to be a positive integer"
        )
        sys.exit(1)