dgroup/lazylead

View on GitHub
lib/lazylead/task/svn/touch.rb

Summary

Maintainability
A
0 mins
Test Coverage
# frozen_string_literal: true

# The MIT License
#
# Copyright (c) 2019-2022 Yurii Dubinka
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"),
# to deal in the Software without restriction, including without limitation
# the rights to use, copy, modify, merge, publish, distribute, sublicense,
# and/or sell copies of the Software, and to permit persons to whom
# the Software is  furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included
# in all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
# ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE
# OR OTHER DEALINGS IN THE SOFTWARE.

require "tmpdir"
require "nokogiri"
require "active_support/core_ext/hash/conversions"
require_relative "../../os"
require_relative "../../salt"
require_relative "../../opts"

module Lazylead
  module Task
    module Svn
      #
      # Send notification about modification of critical files in svn repo.
      #
      class Touch
        def initialize(log = Log.new)
          @log = log
        end

        def run(_, postman, opts)
          commits = touch(opts)
          postman.send(opts.merge(entries: commits)) unless commits.empty?
        end

        # Return all svn commits for a particular date range, which are touching
        #  somehow the critical files within the svn repo.
        def touch(opts)
          files = opts.slice("files", ",")
          branches = opts.slice("branches", ",")
          svn_log(opts)
            .select { |e| branches.empty? || branches.any? { |b| e.paths.path.start_with? b } }
            .each do |e|
            e.paths.path.delete_if { |p| files.none? { |f| p.include? f } } if e.paths.path.respond_to? :delete_if
          end
        end

        # Return svn commits history for file(s) within particular date range in repo
        def svn_log(opts)
          now = if opts.key? "now"
                  DateTime.parse(opts["now"])
                else
                  DateTime.now
                end
          start = (now.to_time - opts["period"].to_i).to_datetime
          locations(opts).flat_map do |file|
            raw = OS.new.run "svn log --no-auth-cache",
                             "--username #{opts.decrypt('svn_user', 'svn_salt')}",
                             "--password #{opts.decrypt('svn_password', 'svn_salt')}",
                             "--xml -v -r {#{start}}:{#{now}} #{opts['svn_url']}/#{file}"
            Nokogiri.XML(raw, nil, "UTF-8")
                    .xpath("//logentry[paths/path]")
                    .map { |xml| to_entry(xml) }
          end
        end

        # Convert single revision(XML text) to entry object.
        # Entry object is a simple ruby struct object.
        def to_entry(xml)
          e = to_struct(Hash.from_xml(xml.to_s.strip)).logentry
          if e.paths.path.respond_to? :each
            e.paths.path.each(&:strip!)
          else
            e.paths.path.strip!
          end
          e
        end

        # Make a simple ruby struct object from hash hierarchically,
        #  considering nested hash(es) (if applicable)
        def to_struct(hsh)
          OpenStruct.new(
            hsh.transform_values { |v| v.is_a?(Hash) ? to_struct(v) : v }
          )
        end

        # Return for each file all his locations considering branches
        #   > locations('readme.md')
        #   branches/0.13.x/readme.md
        #   trunk/readme.md
        #
        # @todo #567/DEV Performance: improve the glob pattern to support multiple files.
        #  Right now in order to find all branches for a particular file we are using glob pattern
        #  - https://en.wikipedia.org/wiki/Glob_(programming)
        #  - https://stackoverflow.com/a/70950401/6916890
        #  - https://globster.xyz
        #  Right now for each file we initiate a separate search request, thus we need to think
        #  how to avoid this and send only one search request
        def locations(opts)
          opts.slice("files", ",").flat_map do |file|
            raw = OS.new.run "svn ls --no-auth-cache ",
                             "--username #{opts.decrypt('svn_user', 'svn_salt')}",
                             "--password #{opts.decrypt('svn_password', 'svn_salt')}",
                             "--xml -R --search \"#{file}\" #{opts['svn_url']}"
            Nokogiri.XML(raw, nil, "UTF-8")
                    .xpath("/lists/list/entry/name/text()")
                    .map { |f| f.to_s.strip }
          end
        end
      end
    end
  end
end