D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
proc
/
self
/
root
/
opt
/
saltstack
/
salt
/
lib
/
python3.10
/
site-packages
/
salt
/
states
/
Filename :
rsync.py
back
Copy
# # Copyright 2015 SUSE LLC # # 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. """ State to synchronize files and directories with rsync. .. versionadded:: 2016.3.0 .. code-block:: yaml /opt/user-backups: rsync.synchronized: - source: /home - force: True """ import logging import os import salt.utils.path log = logging.getLogger(__name__) def __virtual__(): """ Only if Rsync is available. :return: """ if salt.utils.path.which("rsync"): return True return (False, "Command not found: rsync") def _get_summary(rsync_out): """ Get summary from the rsync successful output. """ return "- " + "\n- ".join( [ elm for elm in rsync_out.split("\n\n")[-1].replace(" ", "\n").split("\n") if elm ] ) def _get_changes(rsync_out): """ Get changes from the rsync successful output. """ copied = list() deleted = list() for line in rsync_out.split("\n\n")[0].split("\n")[1:]: if line.startswith("deleting "): deleted.append(line.split(" ", 1)[-1]) else: copied.append(line) ret = { "copied": os.linesep.join(sorted(copied)) or "N/A", "deleted": os.linesep.join(sorted(deleted)) or "N/A", } # Return whether anything really changed ret["changed"] = not ((ret["copied"] == "N/A") and (ret["deleted"] == "N/A")) return ret def synchronized( name, source, delete=False, force=False, update=False, passwordfile=None, exclude=None, excludefrom=None, prepare=False, dryrun=False, additional_opts=None, ): """ Guarantees that the source directory is always copied to the target. name Name of the target directory. source Source directory. prepare Create destination directory if it does not exists. delete Delete extraneous files from the destination dirs (True or False) force Force deletion of dirs even if not empty update Skip files that are newer on the receiver (True or False) passwordfile Read daemon-access password from the file (path) exclude Exclude files, that matches pattern. excludefrom Read exclude patterns from the file (path) dryrun Perform a trial run with no changes made. Is the same as doing test=True .. versionadded:: 2016.3.1 additional_opts Pass additional options to rsync, should be included as a list. .. versionadded:: 2018.3.0 """ ret = {"name": name, "changes": {}, "result": True, "comment": ""} if not os.path.exists(name) and not force and not prepare: ret["result"] = False ret["comment"] = "Destination directory {dest} was not found.".format(dest=name) else: if not os.path.exists(name) and prepare: os.makedirs(name) if __opts__["test"]: dryrun = True result = __salt__["rsync.rsync"]( source, name, delete=delete, force=force, update=update, passwordfile=passwordfile, exclude=exclude, excludefrom=excludefrom, dryrun=dryrun, additional_opts=additional_opts, ) if __opts__["test"] or dryrun: ret["result"] = None ret["comment"] = _get_summary(result["stdout"]) return ret # Failed if result.get("retcode"): ret["result"] = False ret["comment"] = result["stderr"] # Changed elif _get_changes(result["stdout"])["changed"]: ret["comment"] = _get_summary(result["stdout"]) ret["changes"] = _get_changes(result["stdout"]) del ret["changes"]["changed"] # Don't need to print the boolean # Clean else: ret["comment"] = _get_summary(result["stdout"]) ret["changes"] = {} return ret