avm 0.2.0 → 0.3.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/lib/avm/executables.rb +24 -0
- data/lib/avm/files.rb +9 -0
- data/lib/avm/files/appendable.rb +55 -0
- data/lib/avm/files/appendable/file_content.rb +24 -0
- data/lib/avm/files/appendable/plain_directory.rb +25 -0
- data/lib/avm/files/appendable/resource_base.rb +13 -0
- data/lib/avm/files/appendable/tar_output_command.rb +26 -0
- data/lib/avm/files/appendable/templatized_directory.rb +29 -0
- data/lib/avm/files/appender.rb +11 -0
- data/lib/avm/files/deploy.rb +71 -0
- data/lib/avm/files/formatter.rb +90 -0
- data/lib/avm/files/formatter/formats.rb +13 -0
- data/lib/avm/files/formatter/formats/base.rb +62 -0
- data/lib/avm/files/formatter/formats/generic_plain.rb +34 -0
- data/lib/avm/files/formatter/formats/html.rb +45 -0
- data/lib/avm/files/formatter/formats/javascript.rb +23 -0
- data/lib/avm/files/formatter/formats/json.rb +27 -0
- data/lib/avm/files/formatter/formats/php.rb +21 -0
- data/lib/avm/files/formatter/formats/python.rb +21 -0
- data/lib/avm/files/formatter/formats/ruby.rb +22 -0
- data/lib/avm/files/formatter/formats/xml.rb +27 -0
- data/lib/avm/files/formatter/utf8_assert.rb +72 -0
- data/lib/avm/files/info.rb +24 -0
- data/lib/avm/files/rotate.rb +107 -0
- data/lib/avm/git.rb +10 -0
- data/lib/avm/git/auto_commit/commit_info.rb +23 -0
- data/lib/avm/git/auto_commit/rules.rb +31 -0
- data/lib/avm/git/auto_commit/rules/base.rb +39 -0
- data/lib/avm/git/auto_commit/rules/last.rb +19 -0
- data/lib/avm/git/auto_commit/rules/manual.rb +45 -0
- data/lib/avm/git/auto_commit/rules/new.rb +24 -0
- data/lib/avm/git/auto_commit/rules/nth.rb +31 -0
- data/lib/avm/git/auto_commit/rules/unique.rb +21 -0
- data/lib/avm/git/auto_commit_path.rb +28 -0
- data/lib/avm/git/auto_commit_path/ruby.rb +20 -0
- data/lib/avm/git/commit.rb +59 -0
- data/lib/avm/git/commit/class_methods.rb +31 -0
- data/lib/avm/git/commit/deploy.rb +38 -0
- data/lib/avm/git/commit/deploy_methods.rb +19 -0
- data/lib/avm/git/commit/diff_tree_line.rb +32 -0
- data/lib/avm/git/commit/file.rb +46 -0
- data/lib/avm/git/file_auto_fixup.rb +83 -0
- data/lib/avm/git/issue.rb +11 -0
- data/lib/avm/git/issue/complete.rb +51 -0
- data/lib/avm/git/issue/complete/commits.rb +42 -0
- data/lib/avm/git/issue/complete/git_subrepos.rb +23 -0
- data/lib/avm/git/issue/complete/local_branch.rb +54 -0
- data/lib/avm/git/issue/complete/local_tag.rb +39 -0
- data/lib/avm/git/issue/complete/push.rb +54 -0
- data/lib/avm/git/issue/complete/remote.rb +33 -0
- data/lib/avm/git/issue/complete/test.rb +45 -0
- data/lib/avm/git/issue/complete/tracker.rb +28 -0
- data/lib/avm/git/issue/complete/validation.rb +31 -0
- data/lib/avm/git/issue/complete/validations.rb +53 -0
- data/lib/avm/git/issue/complete/working_tree.rb +19 -0
- data/lib/avm/git/issue/deliver.rb +56 -0
- data/lib/avm/git/organize.rb +11 -0
- data/lib/avm/git/organize/reference_update.rb +34 -0
- data/lib/avm/git/organize/repository.rb +76 -0
- data/lib/avm/git/revision_test.rb +106 -0
- data/lib/avm/git/subrepo_check.rb +38 -0
- data/lib/avm/git/subrepo_check/parent.rb +51 -0
- data/lib/avm/git/subrepo_check/remote.rb +89 -0
- data/lib/avm/git/subrepo_check/show_result.rb +32 -0
- data/lib/avm/git/subrepo_checks.rb +59 -0
- data/lib/avm/result.rb +86 -0
- data/lib/avm/version.rb +1 -1
- 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,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
|