avm 0.2.0 → 0.3.0

Sign up to get free protection for your applications and to get access to all the features.
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