avm 0.2.0 → 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (69) hide show
  1. checksums.yaml +4 -4
  2. data/lib/avm/executables.rb +24 -0
  3. data/lib/avm/files.rb +9 -0
  4. data/lib/avm/files/appendable.rb +55 -0
  5. data/lib/avm/files/appendable/file_content.rb +24 -0
  6. data/lib/avm/files/appendable/plain_directory.rb +25 -0
  7. data/lib/avm/files/appendable/resource_base.rb +13 -0
  8. data/lib/avm/files/appendable/tar_output_command.rb +26 -0
  9. data/lib/avm/files/appendable/templatized_directory.rb +29 -0
  10. data/lib/avm/files/appender.rb +11 -0
  11. data/lib/avm/files/deploy.rb +71 -0
  12. data/lib/avm/files/formatter.rb +90 -0
  13. data/lib/avm/files/formatter/formats.rb +13 -0
  14. data/lib/avm/files/formatter/formats/base.rb +62 -0
  15. data/lib/avm/files/formatter/formats/generic_plain.rb +34 -0
  16. data/lib/avm/files/formatter/formats/html.rb +45 -0
  17. data/lib/avm/files/formatter/formats/javascript.rb +23 -0
  18. data/lib/avm/files/formatter/formats/json.rb +27 -0
  19. data/lib/avm/files/formatter/formats/php.rb +21 -0
  20. data/lib/avm/files/formatter/formats/python.rb +21 -0
  21. data/lib/avm/files/formatter/formats/ruby.rb +22 -0
  22. data/lib/avm/files/formatter/formats/xml.rb +27 -0
  23. data/lib/avm/files/formatter/utf8_assert.rb +72 -0
  24. data/lib/avm/files/info.rb +24 -0
  25. data/lib/avm/files/rotate.rb +107 -0
  26. data/lib/avm/git.rb +10 -0
  27. data/lib/avm/git/auto_commit/commit_info.rb +23 -0
  28. data/lib/avm/git/auto_commit/rules.rb +31 -0
  29. data/lib/avm/git/auto_commit/rules/base.rb +39 -0
  30. data/lib/avm/git/auto_commit/rules/last.rb +19 -0
  31. data/lib/avm/git/auto_commit/rules/manual.rb +45 -0
  32. data/lib/avm/git/auto_commit/rules/new.rb +24 -0
  33. data/lib/avm/git/auto_commit/rules/nth.rb +31 -0
  34. data/lib/avm/git/auto_commit/rules/unique.rb +21 -0
  35. data/lib/avm/git/auto_commit_path.rb +28 -0
  36. data/lib/avm/git/auto_commit_path/ruby.rb +20 -0
  37. data/lib/avm/git/commit.rb +59 -0
  38. data/lib/avm/git/commit/class_methods.rb +31 -0
  39. data/lib/avm/git/commit/deploy.rb +38 -0
  40. data/lib/avm/git/commit/deploy_methods.rb +19 -0
  41. data/lib/avm/git/commit/diff_tree_line.rb +32 -0
  42. data/lib/avm/git/commit/file.rb +46 -0
  43. data/lib/avm/git/file_auto_fixup.rb +83 -0
  44. data/lib/avm/git/issue.rb +11 -0
  45. data/lib/avm/git/issue/complete.rb +51 -0
  46. data/lib/avm/git/issue/complete/commits.rb +42 -0
  47. data/lib/avm/git/issue/complete/git_subrepos.rb +23 -0
  48. data/lib/avm/git/issue/complete/local_branch.rb +54 -0
  49. data/lib/avm/git/issue/complete/local_tag.rb +39 -0
  50. data/lib/avm/git/issue/complete/push.rb +54 -0
  51. data/lib/avm/git/issue/complete/remote.rb +33 -0
  52. data/lib/avm/git/issue/complete/test.rb +45 -0
  53. data/lib/avm/git/issue/complete/tracker.rb +28 -0
  54. data/lib/avm/git/issue/complete/validation.rb +31 -0
  55. data/lib/avm/git/issue/complete/validations.rb +53 -0
  56. data/lib/avm/git/issue/complete/working_tree.rb +19 -0
  57. data/lib/avm/git/issue/deliver.rb +56 -0
  58. data/lib/avm/git/organize.rb +11 -0
  59. data/lib/avm/git/organize/reference_update.rb +34 -0
  60. data/lib/avm/git/organize/repository.rb +76 -0
  61. data/lib/avm/git/revision_test.rb +106 -0
  62. data/lib/avm/git/subrepo_check.rb +38 -0
  63. data/lib/avm/git/subrepo_check/parent.rb +51 -0
  64. data/lib/avm/git/subrepo_check/remote.rb +89 -0
  65. data/lib/avm/git/subrepo_check/show_result.rb +32 -0
  66. data/lib/avm/git/subrepo_checks.rb +59 -0
  67. data/lib/avm/result.rb +86 -0
  68. data/lib/avm/version.rb +1 -1
  69. metadata +123 -3
@@ -0,0 +1,32 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Avm
4
+ module Git
5
+ class Commit
6
+ class DiffTreeLine
7
+ DIFF_TREE_PATTERN = /\A:(\d{6}) (\d{6}) (\S+) (\S+) (\S+)\t(\S.*)\z/.freeze
8
+ FIELDS = %w[src_mode dst_mode src_sha1 dst_sha1 status path].freeze
9
+ GIT_COMMAND_ARGS = %w[-c core.quotepath=off diff-tree --no-commit-id -r --full-index].freeze
10
+
11
+ attr_reader(*FIELDS)
12
+
13
+ # line: a line of command "git [GIT_COMMAND_ARGS]"'s output.
14
+ # Reference: https://git-scm.com/docs/git-diff-tree
15
+ def initialize(line)
16
+ m = DIFF_TREE_PATTERN.match(line.strip)
17
+ raise "\"#{line}\" did not match pattern" unless m
18
+
19
+ FIELDS.count.times { |i| send("#{FIELDS[i]}=", m[i + 1]) }
20
+ end
21
+
22
+ def fields
23
+ FIELDS.map { |field| [field, send(field)] }.to_h
24
+ end
25
+
26
+ private
27
+
28
+ attr_writer(*FIELDS)
29
+ end
30
+ end
31
+ end
32
+ end
@@ -0,0 +1,46 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'active_support/core_ext/module/delegation'
4
+ require 'eac_ruby_utils/simple_cache'
5
+ require 'avm/git/commit/diff_tree_line'
6
+
7
+ module Avm
8
+ module Git
9
+ class Commit
10
+ class File
11
+ include ::EacRubyUtils::SimpleCache
12
+
13
+ attr_reader :git, :diff_tree
14
+
15
+ # git: [EacGit::Local]
16
+ # diff_tree_tree: a line of command "git diff-tree --no-commit-id -r --full-index"'s output
17
+ def initialize(git, diff_tree_line)
18
+ @git = git
19
+ @diff_tree = ::Avm::Git::Commit::DiffTreeLine.new(diff_tree_line)
20
+ end
21
+
22
+ delegate(*::Avm::Git::Commit::DiffTreeLine::FIELDS, to: :diff_tree)
23
+
24
+ def to_s
25
+ "#{path}|#{status}"
26
+ end
27
+
28
+ def src_size_uncached
29
+ size(src_sha1)
30
+ end
31
+
32
+ def dst_size_uncached
33
+ size(dst_sha1)
34
+ end
35
+
36
+ private
37
+
38
+ def size(sha1)
39
+ return 0 if /\A0+\z/.match(sha1)
40
+
41
+ git.command('cat-file', '-s', sha1).execute!.strip.to_i
42
+ end
43
+ end
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,83 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'avm/git/auto_commit/commit_info'
4
+ require 'eac_ruby_utils/core_ext'
5
+
6
+ module Avm
7
+ module Git
8
+ class FileAutoFixup
9
+ enable_speaker
10
+ enable_simple_cache
11
+ enable_listable
12
+
13
+ common_constructor :git, :path, :rules do
14
+ self.path = path.to_pathname.expand_path(git.root_path)
15
+ end
16
+
17
+ COMMITS_SEARCH_INTERVAL = 'origin/master..HEAD'
18
+
19
+ def git_relative_path
20
+ path.to_pathname.relative_path_from(git.root_path)
21
+ end
22
+
23
+ def run
24
+ start_banner
25
+ run_commit || warn("No rule returned commit information for \"#{path}\"")
26
+ end
27
+
28
+ private
29
+
30
+ def commit_args
31
+ commit_info.if_present([], &:git_commit_args) + ['--', git_relative_path]
32
+ end
33
+
34
+ def commit_info_uncached
35
+ rules.lazy.map { |rule| rule.with_file(self).commit_info }.find(&:present?)
36
+ end
37
+
38
+ def start_banner
39
+ infov 'Path', path
40
+ infov ' Commits found', commits.count
41
+ end
42
+
43
+ def run_commit
44
+ return false if commit_info.blank?
45
+
46
+ infov ' Commit arguments', ::Shellwords.join(commit_args)
47
+ run_git_add_and_commit
48
+ success ' Commited'
49
+ true
50
+ end
51
+
52
+ def run_git_add_and_commit
53
+ git.execute!('reset', '--soft', 'HEAD')
54
+ if path.exist?
55
+ git.execute!('add', git_relative_path)
56
+ else
57
+ git.execute!('rm', '-f', git_relative_path)
58
+ end
59
+ git.execute!('commit', *commit_args)
60
+ end
61
+
62
+ def commits_uncached
63
+ git.execute!('log', '--pretty=format:%H', COMMITS_SEARCH_INTERVAL, '--', path)
64
+ .each_line.map { |sha1| ::Avm::Git::Commit.new(git, sha1.strip) }
65
+ .reject { |commit| commit.subject.start_with?('fixup!') }
66
+ .each_with_index.map { |commit, index| CommitDelegator.new(commit, index) }
67
+ end
68
+
69
+ class CommitDelegator < ::SimpleDelegator
70
+ attr_reader :index
71
+
72
+ def initialize(commit, index)
73
+ super(commit)
74
+ @index = index
75
+ end
76
+
77
+ def position
78
+ index + 1
79
+ end
80
+ end
81
+ end
82
+ end
83
+ end
@@ -0,0 +1,11 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'eac_ruby_utils/require_sub'
4
+ ::EacRubyUtils.require_sub(__FILE__)
5
+
6
+ module Avm
7
+ module Git
8
+ module Issue
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,51 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'avm/tools/core_ext'
4
+
5
+ module Avm
6
+ module Git
7
+ module Issue
8
+ class Complete
9
+ require_sub __FILE__, include_modules: true
10
+ enable_simple_cache
11
+ enable_speaker
12
+
13
+ attr_reader :skip_validations
14
+
15
+ def initialize(options)
16
+ consumer = ::EacRubyUtils::OptionsConsumer.new(options)
17
+ dir, @skip_validations = consumer.consume_all(:dir, :skip_validations)
18
+ validate_skip_validations
19
+ consumer.validate
20
+ @git = ::Avm::Launcher::Git::Base.new(dir)
21
+ end
22
+
23
+ def start_banner
24
+ validations_banner
25
+ end
26
+
27
+ def run
28
+ return false unless valid?
29
+
30
+ assert_tag
31
+ push
32
+ remove_local_branch
33
+ clipboard_copy_tracker_message
34
+ true
35
+ end
36
+
37
+ def issue_id
38
+ m = branch_name.match(/\A#{Regexp.quote('issue_')}(\d+)\z/)
39
+ m ? m[1].to_i : nil
40
+ end
41
+
42
+ private
43
+
44
+ def git(args, exit_outputs = {})
45
+ r = @git.execute!(args, exit_outputs: exit_outputs)
46
+ r.is_a?(String) ? r.strip : r
47
+ end
48
+ end
49
+ end
50
+ end
51
+ end
@@ -0,0 +1,42 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'avm/result'
4
+
5
+ module Avm
6
+ module Git
7
+ module Issue
8
+ class Complete
9
+ module Commits
10
+ def commits_result
11
+ ::Avm::Result.success_or_error(commits.any?, 'yes', 'none')
12
+ end
13
+
14
+ def commits_uncached
15
+ return [] unless branch_hash && follow_master?
16
+
17
+ interval = remote_master_hash ? "#{remote_master_hash}..#{branch_hash}" : branch_hash
18
+ @git.execute!('rev-list', interval).each_line.map(&:strip)
19
+ end
20
+
21
+ def bifurcations_result
22
+ commits.each do |commit|
23
+ if multiple_parents?(commit)
24
+ return ::Avm::Result.error("#{commit} has multiple parents")
25
+ end
26
+ end
27
+ ::Avm::Result.success('no')
28
+ end
29
+
30
+ def multiple_parents?(commit)
31
+ commit_parents(commit).count > 1
32
+ end
33
+
34
+ def commit_parents(commit)
35
+ @git.execute!('log', '--pretty=%P', '-n', '1', commit).split(' ').map(&:strip)
36
+ .select(&:present?)
37
+ end
38
+ end
39
+ end
40
+ end
41
+ end
42
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'avm/git/subrepo_checks'
4
+ require 'eac_git/local'
5
+
6
+ module Avm
7
+ module Git
8
+ module Issue
9
+ class Complete
10
+ module GitSubrepos
11
+ def git_subrepos_result
12
+ return ::Avm::Result.error('Unclean workspace') unless clean_workspace?
13
+
14
+ infom 'Checking Git subrepos...'
15
+ r = ::Avm::Git::SubrepoChecks.new(::EacGit::Local.new(@git)).add_all_subrepos
16
+ r.check_remote = true
17
+ r.result
18
+ end
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,54 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'avm/result'
4
+
5
+ module Avm
6
+ module Git
7
+ module Issue
8
+ class Complete
9
+ module LocalBranch
10
+ def branch_uncached
11
+ @git.current_branch
12
+ end
13
+
14
+ def branch_hash_uncached
15
+ @git.rev_parse("refs/heads/#{branch}")
16
+ end
17
+
18
+ def branch_name
19
+ branch.split('/')[-1]
20
+ end
21
+
22
+ def branch_name_result
23
+ ::Avm::Result.success_or_error(issue_id.present?, branch_name)
24
+ end
25
+
26
+ def branch_hash_result
27
+ ::Avm::Result.success_or_error(
28
+ branch_hash.present?,
29
+ branch_hash
30
+ )
31
+ end
32
+
33
+ def follow_master_result
34
+ return ::Avm::Result.neutral('No branch hash') unless branch_hash
35
+
36
+ r = follow_master?
37
+ ::Avm::Result.success_or_error(r, 'yes', 'no')
38
+ end
39
+
40
+ def follow_master?
41
+ remote_master_hash ? @git.descendant?(branch_hash, remote_master_hash) : true
42
+ end
43
+
44
+ def remove_local_branch
45
+ info 'Removendo branch local...'
46
+ bn = branch_name
47
+ git(['checkout', branch_hash])
48
+ git(['branch', '-D', bn])
49
+ end
50
+ end
51
+ end
52
+ end
53
+ end
54
+ end
@@ -0,0 +1,39 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'avm/result'
4
+
5
+ module Avm
6
+ module Git
7
+ module Issue
8
+ class Complete
9
+ module LocalTag
10
+ def assert_tag
11
+ if tag_hash
12
+ return if tag_hash == branch_hash
13
+
14
+ delete_tag
15
+ end
16
+ create_tag
17
+ end
18
+
19
+ def delete_tag
20
+ info 'Removendo tag...'
21
+ git(['tag', '-d', branch_name])
22
+ end
23
+
24
+ def tag
25
+ "refs/tags/#{branch_name}"
26
+ end
27
+
28
+ def tag_hash
29
+ @git.rev_parse(tag)
30
+ end
31
+
32
+ def create_tag
33
+ git(['tag', branch_name, branch_hash])
34
+ end
35
+ end
36
+ end
37
+ end
38
+ end
39
+ end
@@ -0,0 +1,54 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Avm
4
+ module Git
5
+ module Issue
6
+ class Complete
7
+ module Push
8
+ def dry_push_args
9
+ %w[push --dry-run] + [remote_name] + pushs
10
+ end
11
+
12
+ def dry_push_result
13
+ return ::Avm::Result.error('Nothing to push') if pushs.empty?
14
+
15
+ r = @git.execute(dry_push_args)
16
+ message = if r.fetch(:exit_code).zero?
17
+ 'ok'
18
+ else
19
+ r.fetch(:stderr) + "\n#{::Shellwords.join(dry_push_args)}"
20
+ end
21
+ ::Avm::Result.success_or_error(r.fetch(:exit_code).zero?, message)
22
+ end
23
+
24
+ def push
25
+ if pushs.empty?
26
+ info 'PUSH: Nada a enviar'
27
+ else
28
+ info "PUSH: enviando \"#{pushs}\"..."
29
+ git(%w[push origin] + pushs)
30
+ end
31
+ end
32
+
33
+ def pushs_uncached
34
+ [master_push, remove_branch_push, tag_push].reject(&:nil?)
35
+ end
36
+
37
+ def master_push
38
+ remote_master_hash != branch_hash ? "#{branch_hash}:refs/heads/master" : nil
39
+ end
40
+
41
+ def remove_branch_push
42
+ remote_branch_hash ? ":refs/heads/#{branch}" : nil
43
+ end
44
+
45
+ def tag_push
46
+ return nil unless !remote_tag_hash || remote_tag_hash != branch_hash
47
+
48
+ "#{branch_hash}:#{tag}"
49
+ end
50
+ end
51
+ end
52
+ end
53
+ end
54
+ end