require 'thor' require 'snapsync' require 'set' require_relative 'tasks/AutoSync' module Snapsync class CLI < Thor class_option :debug, type: :boolean, default: false class_option :ssh_debug, type: :boolean, default: false class_option :systemd, type: :boolean, default: false, desc: "Tells snapsync it is being run from a systemd service" class_option :notify, type: :boolean, default: false, desc: "Use libnotify to notify all users in system" no_commands do def config_from_name(name) path = Pathname.new(name) if !path.exist? path = SnapperConfig.default_config_dir + path if !path.exist? raise ArgumentError, "cannot find any snapper configuration called #{name}" end end SnapperConfig.load(path) end def handle_class_options if options[:debug] Snapsync.logger.level = 'DEBUG' end Snapsync.SSH_DEBUG = options[:ssh_debug] Snapsync.SYSTEMD = options[:systemd] Snapsync.NOTIFY = options[:notify] end # Resolves a path (or nil) into a list of snapsync targets and # yields them # # @param [AgnosticPath,nil] dir the path the user gave, or nil if all # available auto-sync paths should be processed. If the directory is # a target, it is yield as-is. It can also be the root of a sync-all # target (with proper snapsync target as subdirectories whose name # matches the snapper configurations) # # @yieldparam [SnapperConfig] config # @yieldparam [SyncTarget] target def each_target(dir = nil) return enum_for(__method__) if !block_given? if dir dir = Snapsync::path(dir) begin return yield(nil, SyncTarget.new(dir, create_if_needed: false)) rescue SyncTarget::InvalidTargetPath end SyncAll.new(dir).each_target do |config, target| yield(config, target) end else autosync = AutoSync.new autosync.each_available_target do |config, target| yield(config, target) end end end # @return [String, Snapsync::Path, Pathname] uuid, mountpoint, relative def partition_of(dir) PartitionsMonitor.new(dir).partition_of(dir) end end desc 'sync ', 'synchronizes the snapper configuration CONFIG with the snapsync target DIR' option :autoclean, type: :boolean, default: nil, desc: 'whether the target should be cleaned of obsolete snapshots', long_desc: "The default is to use the value specified in the target's configuration file. This command line option allows to override the default" def sync(config_name, dir) handle_class_options dir = Snapsync::path(dir) config = config_from_name(config_name) target = SyncTarget.new(dir) Sync.new(config, target, autoclean: options[:autoclean]).run end desc 'sync-all ', 'synchronizes all snapper configurations into corresponding subdirectories of DIR' option :autoclean, type: :boolean, default: nil, desc: 'whether the target should be cleaned of obsolete snapshots', long_desc: "The default is to use the value specified in the target's configuration file. This command line option allows to override the default" def sync_all(dir) handle_class_options dir = Snapsync::path(dir) op = SyncAll.new(dir, config_dir: SnapperConfig.default_config_dir, autoclean: options[:autoclean]) op.run end desc 'cleanup [--debug] [--dry-run] ', 'cleans up the snapsync target DIR based on the policy set by the policy command' option :dry_run, type: :boolean, default: false def cleanup(dir) handle_class_options target = SyncTarget.new(Snapsync::path(dir)) if target.cleanup target.cleanup.cleanup(target, dry_run: options[:dry_run]) else Snapsync.info "#{target.sync_policy.class.name} policy set, nothing to do" end end no_commands do def normalize_policy(args) policy = if args.empty? ['default', Array.new] elsif args.size == 1 args + [Array.new] else [args.shift, args] end SyncTarget.parse_policy(*policy) return *policy end end desc 'init [NAME] [POLICY]', 'creates a synchronization target, optionally adding it to the auto-sync targets and specifying the synchronization and cleanup policies' long_desc <<-EOD NAME must be provided if DIR is to be added to the auto-sync targets (which is the default). DIR can be a remote filesystem path in scp-like format ( [user[:password]@]host:/path/to/drive/snapsync ) By default, the default policy is used. To change this, provide additional arguments as would be expected by the policy subcommand. Run snapsync help policy for more information EOD option :all, type: :boolean, default: true, desc: "if true (the default), create one snapsync target per snapper configuration under DIR, otherwise, initialize only one target directly in DIR" option :auto, type: :boolean, default: true, desc: "if true (the default), add the newly created target to auto-sync" option :automount, type: :boolean, default: true, desc: 'whether the supporting partition should be auto-mounted by snapsync when needed or not (the default is yes). Only useful if --no-auto has not been provided on the command line.' option :config_file, default: '/etc/snapsync.conf', desc: 'the configuration file that should be updated' def init(*args) if options[:auto] && !options[:all] raise ArgumentError, "cannot use --auto without --all" end if options[:auto] if args.size < 2 self.class.handle_argument_error(self.class.all_commands['init'], nil, args, 2) end name, dir, *policy = *args else if args.size < 1 self.class.handle_argument_error(self.class.all_commands['init'], nil, args, 1) end dir, *policy = *args end dir = Snapsync::path(dir) remote = dir.instance_of? RemotePathname # Parse the policy option early to avoid breaking later begin policy = normalize_policy(policy) rescue Exception => policy_validation_error # Try to see if the user forgot to add the NAME option or added # the name option but should not have if (args.size > 1) && options[:auto] begin normalize_policy(args[1..-1]) raise ArgumentError, "--auto is set but it seems that you did not provide a name" rescue InvalidConfiguration end elsif args.size > 2 begin normalize_policy(args[2..-1]) raise ArgumentError, "--auto is not set but it seems that you provided a name" rescue InvalidConfiguration end end raise policy_validation_error end dirs = Array.new if options[:all] SnapperConfig.each_in_dir do |config| dirs << dir + config.name end else dirs << dir end dirs.each do |path| begin SyncTarget.new(path, create_if_needed: false) Snapsync.info "#{path} was already initialized" rescue ArgumentError, SyncTarget::NoUUIDError path.mkpath target = SyncTarget.new(path) target.change_policy(*policy) target.write_config Snapsync.info "initialized #{path} as a snapsync target" end end # We check that both options are set together for some added safety, # but it's checked at the top of the method if options[:auto] && options[:all] auto_add(name, dir) end end desc 'auto-add [NAME] ', "add DIR to the set of targets for auto-sync" option :automount, type: :boolean, default: true, desc: 'whether the supporting partition should be auto-mounted by snapsync when needed or not (the default is yes)' option :config_file, default: '/etc/snapsync.conf', desc: 'the configuration file that should be updated' def auto_add(name, dir) uuid, mountpoint, relative = partition_of(Snapsync::path(dir)) conf_path = Pathname.new(options[:config_file]) autosync = AutoSync.new snapsync_config_file: conf_path exists = autosync.each_autosync_target.find do |t| t.partition_uuid == uuid && t.mountpoint.cleanpath == mountpoint.cleanpath && t.relative.cleanpath == relative.cleanpath end if exists if !exists.name if (exists.automount ^ options[:automount]) && name Snapsync.info "already exists without a name, setting the name to #{name}" elsif name Snapsync.info "already exists without a name and a different automount flag, setting the name to #{name} and updating the automount flag" else Snapsync.info "already exists with different automount flag, updating" end elsif exists.automount == options[:automount] Snapsync.info "already exists under the name #{exists.name}" else Snapsync.info "already exists under the name #{exists.name} but with a different automount flag, changing" exists.automount = options[:automount] end exists.name ||= name else autosync.add AutoSync::AutoSyncTarget.new(uuid, mountpoint, relative, options[:automount], name) end autosync.write_config(conf_path) end desc 'auto-remove NAME', "remove a target from auto-sync by name" def auto_remove(name) conf_path = Pathname.new('/etc/snapsync.conf') autosync = AutoSync.new snapsync_config_file: conf_path autosync.remove(name: name) autosync.write_config(conf_path) end desc 'policy [OPTIONS]', 'sets the synchronization and cleanup policy for the given target or targets' long_desc <<-EOD This command sets the policy used to decide which snapshots to synchronize to the target, and which to not synchronize. TYPE: Three policy types can be used: default, last, or timeline The 'default' policy takes no argument. It will synchronize all snapshots present in the source, and do no cleanup The 'last' policy takes no argument. It will synchronize (and keep) only the last snapshot The 'timeline' policy takes periods of time as argument (as e.g. day 10 or month 20). It will keep at least one snapshot for each period, and for the duration specified (day 10 tells to keep one snapshot per day for 10 days). snapsync understands the following period names: year, month, week, day, hour. OPTIONS := { year {int} | month {int} | week {int} | day {int} | hour {int} } EOD def policy(dir, type, *options) handle_class_options dir = Snapsync::path(dir) # Parse the policy early to avoid breaking later policy = normalize_policy([type, *options]) each_target(dir) do |_, target| target.change_policy(*policy) target.write_config end end desc 'destroy ', 'destroys a snapsync target' long_desc <<-EOD While it can easily be done manually, this command makes sure that the snapshots are properly deleted EOD def destroy(dir) handle_class_options target_dir = Snapsync::path(dir) target = SyncTarget.new(target_dir, create_if_needed: false) snapshots = target.each_snapshot.to_a snapshots.sort_by(&:num).each do |s| target.delete(s) end target_dir.rmtree end desc "auto-sync", "automatic synchronization" option :one_shot, desc: "do one synchronization and quit", type: :boolean, default: false option :config_file, desc: "path to the config file (defaults to /etc/snapsync.conf)", default: '/etc/snapsync.conf' option :local_only, desc: "Only synchronize to local targets", type: :boolean, default: false option :remote_only, desc: "Only synchronize to remote targets", type: :boolean, default: false def auto_sync handle_class_options if options[:local_only] and options[:remote_only] raise ArgumentError, '--local-only and --remote-only cannot be used together' end while true auto = AutoSync.new(SnapperConfig.default_config_dir, snapsync_config_file: Pathname.new(options[:config_file]), sync_local: options[:local_only] || !options[:remote_only], sync_remote: options[:remote_only] || !options[:local_only]) auto.run break if options[:one_shot] Snapsync.info "done all declared autosync partitions, sleeping #{period}s" sleep 600 end end desc 'list [DIR]', 'list the snapshots present on DIR. If DIR is omitted, tries to access all targets defined as auto-sync targets' def list(dir = nil) handle_class_options each_target(dir) do |config, target| puts "== #{target.dir}" puts "UUID: #{target.uuid}" puts "Enabled: #{target.enabled?}" puts "Autoclean: #{target.autoclean?}" puts "Snapper config: #{config.name}" print "Policy: " pp target.sync_policy snapshots_seen = Set.new # @type [Snapshot] last_snapshot = nil puts "Snapshots:" target.each_snapshot do |s| snapshots_seen.add(s.num) last_snapshot = s puts " #{s.num} #{s.to_time}" end puts " [transferrable:]" config.each_snapshot do |s| if not snapshots_seen.include? s.num delta = s.size_diff_from(last_snapshot) puts " #{s.num} #{s.to_time} => from: #{last_snapshot.num} delta: " \ +"#{Snapsync.human_readable_size(delta)}" # If the delta was 0, then the data already exists on remote. if delta > 0 last_snapshot = s end end end end end end end