ruby_git 0.1.3 → 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.
- checksums.yaml +4 -4
- data/.commitlintrc.yml +16 -0
- data/.github/CODEOWNERS +4 -0
- data/.github/workflows/continuous_integration.yml +87 -0
- data/.github/workflows/enforce_conventional_commits.yml +21 -0
- data/.github/workflows/experimental_ruby_builds.yml +65 -0
- data/.gitignore +3 -0
- data/.husky/commit-msg +1 -0
- data/.markdownlint.yml +25 -0
- data/.rubocop.yml +13 -15
- data/.yardopts +6 -1
- data/CHANGELOG.md +76 -20
- data/CONTRIBUTING.md +7 -7
- data/{LICENSE.md → LICENSE.txt} +1 -1
- data/PLAN.md +67 -0
- data/README.md +64 -10
- data/RELEASING.md +5 -54
- data/Rakefile +31 -38
- data/RubyGit Class Diagram.svg +1 -0
- data/bin/command-line-test +189 -0
- data/bin/console +2 -0
- data/bin/setup +13 -2
- data/lib/ruby_git/command_line/options.rb +61 -0
- data/lib/ruby_git/command_line/result.rb +155 -0
- data/lib/ruby_git/command_line/runner.rb +296 -0
- data/lib/ruby_git/command_line.rb +95 -0
- data/lib/ruby_git/encoding_normalizer.rb +49 -0
- data/lib/ruby_git/errors.rb +169 -0
- data/lib/ruby_git/repository.rb +33 -0
- data/lib/ruby_git/status/branch.rb +92 -0
- data/lib/ruby_git/status/entry.rb +162 -0
- data/lib/ruby_git/status/ignored_entry.rb +44 -0
- data/lib/ruby_git/status/ordinary_entry.rb +207 -0
- data/lib/ruby_git/status/parser.rb +203 -0
- data/lib/ruby_git/status/renamed_entry.rb +257 -0
- data/lib/ruby_git/status/report.rb +143 -0
- data/lib/ruby_git/status/stash.rb +33 -0
- data/lib/ruby_git/status/submodule_status.rb +85 -0
- data/lib/ruby_git/status/unmerged_entry.rb +248 -0
- data/lib/ruby_git/status/untracked_entry.rb +52 -0
- data/lib/ruby_git/status.rb +33 -0
- data/lib/ruby_git/version.rb +1 -1
- data/lib/ruby_git/worktree.rb +277 -0
- data/lib/ruby_git.rb +154 -14
- data/package.json +11 -0
- data/ruby_git.gemspec +32 -20
- metadata +146 -45
- data/.travis.yml +0 -13
- data/MAINTAINERS.md +0 -8
- data/lib/ruby_git/file_helpers.rb +0 -42
- data/lib/ruby_git/git_binary.rb +0 -90
- /data/{ISSUE_TEMPLATE.md → .github/ISSUE_TEMPLATE.md} +0 -0
- /data/{PULL_REQUEST_TEMPLATE.md → .github/PULL_REQUEST_TEMPLATE.md} +0 -0
@@ -0,0 +1,248 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative 'entry'
|
4
|
+
|
5
|
+
module RubyGit
|
6
|
+
module Status
|
7
|
+
# Represents an unmerged file in git status
|
8
|
+
#
|
9
|
+
# @api public
|
10
|
+
class UnmergedEntry < Entry
|
11
|
+
# @attribute [r] conflict_type
|
12
|
+
#
|
13
|
+
# The type of merge conflict
|
14
|
+
#
|
15
|
+
# @example
|
16
|
+
# entry.conflict_type #=> :both_deleted
|
17
|
+
#
|
18
|
+
# @see RubyGit::Status::UnmergedEntry::CONFLICT_TYPES
|
19
|
+
#
|
20
|
+
# @return [Symbol]
|
21
|
+
#
|
22
|
+
# @api public
|
23
|
+
attr_reader :conflict_type
|
24
|
+
|
25
|
+
# @attribute [r] submodule_status
|
26
|
+
#
|
27
|
+
# The submodule status if the entry is a submodule or nil
|
28
|
+
#
|
29
|
+
# @example
|
30
|
+
# entry.submodule #=> 'N...'
|
31
|
+
#
|
32
|
+
# @return [SubmoduleStatus, nil]
|
33
|
+
#
|
34
|
+
# @api public
|
35
|
+
attr_reader :submodule_status
|
36
|
+
|
37
|
+
# @attribute [r] base_mode
|
38
|
+
#
|
39
|
+
# The mode of the file in the base
|
40
|
+
#
|
41
|
+
# @example
|
42
|
+
# entry.base_mode #=> 0o100644
|
43
|
+
#
|
44
|
+
# @return [Integer]
|
45
|
+
#
|
46
|
+
# @api public
|
47
|
+
attr_reader :base_mode
|
48
|
+
|
49
|
+
# @attribute [r] our_mode
|
50
|
+
#
|
51
|
+
# The mode of the file in our branch
|
52
|
+
#
|
53
|
+
# @example
|
54
|
+
# entry.our_mode #=> 0o100644
|
55
|
+
#
|
56
|
+
# @return [Integer]
|
57
|
+
#
|
58
|
+
# @api public
|
59
|
+
attr_reader :our_mode
|
60
|
+
|
61
|
+
# @attribute [r] their_mode
|
62
|
+
#
|
63
|
+
# The mode of the file in their branch
|
64
|
+
#
|
65
|
+
# @example
|
66
|
+
# entry.their_mode #=> 0o100644
|
67
|
+
#
|
68
|
+
# @return [Integer]
|
69
|
+
#
|
70
|
+
# @api public
|
71
|
+
attr_reader :their_mode
|
72
|
+
|
73
|
+
# @attribute [r] worktree_mode
|
74
|
+
#
|
75
|
+
# The mode of the file in the worktree
|
76
|
+
#
|
77
|
+
# @example
|
78
|
+
# entry.worktree_mode #=> 0o100644
|
79
|
+
#
|
80
|
+
# @return [Integer]
|
81
|
+
#
|
82
|
+
# @api public
|
83
|
+
attr_reader :worktree_mode
|
84
|
+
|
85
|
+
# @attribute [r] base_sha
|
86
|
+
#
|
87
|
+
# The SHA of the file in the base
|
88
|
+
#
|
89
|
+
# @example
|
90
|
+
# entry.base_sha #=> 'd670460b4b4aece5915caf5c68d12f560a9fe3e4'
|
91
|
+
#
|
92
|
+
# @return [String]
|
93
|
+
#
|
94
|
+
# @api public
|
95
|
+
attr_reader :base_sha
|
96
|
+
|
97
|
+
# @attribute [r] our_sha
|
98
|
+
#
|
99
|
+
# The SHA of the file in our branch
|
100
|
+
#
|
101
|
+
# @example
|
102
|
+
# entry.our_sha #=> 'd670460b4b4aece5915caf5c68d12f560a9fe3e4'
|
103
|
+
#
|
104
|
+
# @return [String]
|
105
|
+
#
|
106
|
+
# @api public
|
107
|
+
attr_reader :our_sha
|
108
|
+
|
109
|
+
# @attribute [r] their_sha
|
110
|
+
#
|
111
|
+
# The SHA of the file in their branch
|
112
|
+
#
|
113
|
+
# @example
|
114
|
+
# entry.their_sha #=> 'd670460b4b4aece5915caf5c68d12f560a9fe3e4'
|
115
|
+
#
|
116
|
+
# @return [String]
|
117
|
+
#
|
118
|
+
# @api public
|
119
|
+
attr_reader :their_sha
|
120
|
+
|
121
|
+
# @attribute [r] path
|
122
|
+
#
|
123
|
+
# The path of the file
|
124
|
+
#
|
125
|
+
# @example
|
126
|
+
# entry.path #=> 'lib/example.rb'
|
127
|
+
#
|
128
|
+
# @return [String]
|
129
|
+
#
|
130
|
+
# @api public
|
131
|
+
attr_reader :path
|
132
|
+
|
133
|
+
# Parse an unmerged change line of git status output
|
134
|
+
#
|
135
|
+
# The line is expected to be in porcelain v2 format with NUL terminators.
|
136
|
+
#
|
137
|
+
# The format is as follows:
|
138
|
+
# u <XY> <sub> <m1> <m2> <m3> <mW> <h1> <h2> <h3> <path>
|
139
|
+
#
|
140
|
+
# @example
|
141
|
+
# line = 'uU N... 100644 100644 100644 100644 d670460b4b4aece5915caf5c68d12f560a9fe3e4 ' \
|
142
|
+
# 'd670460b4b4aece5915caf5c68d12f560a9fe3e4 d670460b4b4aece5915caf5c68d12f560a9fe3e4 lib/example.rb'
|
143
|
+
# UnmergedEntry.parse(line) #=> #<RubyGit::Status::UnmergedEntry:0x00000001046bd488 ...>
|
144
|
+
#
|
145
|
+
# @param line [String] line from git status
|
146
|
+
#
|
147
|
+
# @return [RubyGit::Status::UnmergedEntry] parsed entry
|
148
|
+
#
|
149
|
+
def self.parse(line) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
|
150
|
+
tokens = line.split(' ', 11)
|
151
|
+
|
152
|
+
new(
|
153
|
+
conflict_type: conflict_code_to_type(tokens[1]),
|
154
|
+
submodule_status: SubmoduleStatus.parse(tokens[2]),
|
155
|
+
base_mode: Integer(tokens[3], 8),
|
156
|
+
our_mode: Integer(tokens[4], 8),
|
157
|
+
their_mode: Integer(tokens[5], 8),
|
158
|
+
worktree_mode: Integer(tokens[6], 8),
|
159
|
+
base_sha: tokens[7],
|
160
|
+
our_sha: tokens[8],
|
161
|
+
their_sha: tokens[9],
|
162
|
+
path: tokens[10]
|
163
|
+
)
|
164
|
+
end
|
165
|
+
|
166
|
+
# Maps the change code to a conflict type symbol
|
167
|
+
CONFLICT_TYPES = {
|
168
|
+
'DD' => :both_deleted,
|
169
|
+
'AU' => :added_by_us,
|
170
|
+
'UD' => :deleted_by_them,
|
171
|
+
'UA' => :added_by_them,
|
172
|
+
'DU' => :deleted_by_us,
|
173
|
+
'AA' => :both_added,
|
174
|
+
'UU' => :both_modified
|
175
|
+
}.freeze
|
176
|
+
|
177
|
+
# Convert conflict code to a symbol
|
178
|
+
#
|
179
|
+
# @example
|
180
|
+
# UnmergedEntry.conflict_code_to_type('DD') #=> :both_deleted
|
181
|
+
#
|
182
|
+
# @param code [String] conflict code
|
183
|
+
# @return [Symbol] conflict type as symbol
|
184
|
+
#
|
185
|
+
def self.conflict_code_to_type(code)
|
186
|
+
CONFLICT_TYPES[code] || :unknown
|
187
|
+
end
|
188
|
+
|
189
|
+
# Initialize a new unmerged entry
|
190
|
+
#
|
191
|
+
# @example
|
192
|
+
# UnmergedEntry.new(
|
193
|
+
# conflict_type: :both_deleted,
|
194
|
+
# submodule_status: nil,
|
195
|
+
# base_mode: 0o100644,
|
196
|
+
# our_mode: 0o100644,
|
197
|
+
# their_mode: 0o100644,
|
198
|
+
# worktree_mode: 0o100644,
|
199
|
+
# base_sha: 'd670460b4b4aece5915caf5c68d12f560a9fe3e4',
|
200
|
+
# our_sha: 'd670460b4b4aece5915caf5c68d12f560a9fe3e4',
|
201
|
+
# their_sha: 'd670460b4b4aece5915caf5c68d12f560a9fe3e4',
|
202
|
+
# path: 'lib/example.rb'
|
203
|
+
# )
|
204
|
+
#
|
205
|
+
# @param conflict_type [Symbol] type of merge conflict
|
206
|
+
# @param submodule_status [SubmoduleStatus, nil] submodule status if applicable
|
207
|
+
# @param base_mode [Integer] mode of the file in the base
|
208
|
+
# @param our_mode [Integer] mode of the file in our branch
|
209
|
+
# @param their_mode [Integer] mode of the file in their branch
|
210
|
+
# @param worktree_mode [Integer] mode of the file in the worktree
|
211
|
+
# @param base_sha [String] SHA of the file in the base
|
212
|
+
# @param our_sha [String] SHA of the file in our branch
|
213
|
+
# @param their_sha [String] SHA of the file in their branch
|
214
|
+
# @param path [String] file path
|
215
|
+
#
|
216
|
+
def initialize( # rubocop:disable Metrics/MethodLength, Metrics/ParameterLists
|
217
|
+
conflict_type:,
|
218
|
+
submodule_status:,
|
219
|
+
base_mode:, our_mode:, their_mode:, worktree_mode:,
|
220
|
+
base_sha:, our_sha:, their_sha:,
|
221
|
+
path:
|
222
|
+
)
|
223
|
+
super(path)
|
224
|
+
@conflict_type = conflict_type
|
225
|
+
@submodule_status = submodule_status
|
226
|
+
@base_mode = base_mode
|
227
|
+
@our_mode = our_mode
|
228
|
+
@their_mode = their_mode
|
229
|
+
@worktree_mode = worktree_mode
|
230
|
+
@base_sha = base_sha
|
231
|
+
@our_sha = our_sha
|
232
|
+
@their_sha = their_sha
|
233
|
+
@path = path
|
234
|
+
end
|
235
|
+
|
236
|
+
# Does the entry represent a merge conflict?
|
237
|
+
#
|
238
|
+
# * Merge conflicts are not considered untracked, staged or unstaged
|
239
|
+
#
|
240
|
+
# @example
|
241
|
+
# entry.conflict? #=> false
|
242
|
+
#
|
243
|
+
# @return [Boolean]
|
244
|
+
#
|
245
|
+
def unmerged? = true
|
246
|
+
end
|
247
|
+
end
|
248
|
+
end
|
@@ -0,0 +1,52 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative 'entry'
|
4
|
+
|
5
|
+
module RubyGit
|
6
|
+
module Status
|
7
|
+
# Represents an untracked file in git status
|
8
|
+
#
|
9
|
+
# @api public
|
10
|
+
class UntrackedEntry < Entry
|
11
|
+
# Parse a git status line to create an untracked entry
|
12
|
+
#
|
13
|
+
# @example
|
14
|
+
# UntrackedEntry.parse('?? lib/example.rb') #=> #<RubyGit::Status::UntrackedEntry:0x00000001046bd488 ...>
|
15
|
+
#
|
16
|
+
# @param line [String] line from git status
|
17
|
+
# @return [RubyGit::Status::UntrackedEntry] parsed entry
|
18
|
+
#
|
19
|
+
def self.parse(line)
|
20
|
+
tokens = line.split(' ', 2)
|
21
|
+
new(path: tokens[1])
|
22
|
+
end
|
23
|
+
|
24
|
+
# Initialize with the path
|
25
|
+
#
|
26
|
+
# @example
|
27
|
+
# UntrackedEntry.new(path: 'file.txt')
|
28
|
+
#
|
29
|
+
# @param path [String] the path of the untracked file
|
30
|
+
#
|
31
|
+
def initialize(path:)
|
32
|
+
super(path)
|
33
|
+
end
|
34
|
+
|
35
|
+
# Is the entry an untracked file?
|
36
|
+
# @example
|
37
|
+
# entry.ignored? #=> false
|
38
|
+
# @return [Boolean]
|
39
|
+
def untracked? = true
|
40
|
+
|
41
|
+
# Does the entry have unstaged changes in the worktree?
|
42
|
+
#
|
43
|
+
# * An entry can have both staged and unstaged changes
|
44
|
+
# * All untracked entries are considered unstaged
|
45
|
+
#
|
46
|
+
# @example
|
47
|
+
# entry.ignored? #=> false
|
48
|
+
# @return [Boolean]
|
49
|
+
def unstaged? = true
|
50
|
+
end
|
51
|
+
end
|
52
|
+
end
|
@@ -0,0 +1,33 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative 'status/branch'
|
4
|
+
require_relative 'status/entry'
|
5
|
+
require_relative 'status/ignored_entry'
|
6
|
+
require_relative 'status/ordinary_entry'
|
7
|
+
require_relative 'status/parser'
|
8
|
+
require_relative 'status/renamed_entry'
|
9
|
+
require_relative 'status/report'
|
10
|
+
require_relative 'status/stash'
|
11
|
+
require_relative 'status/submodule_status'
|
12
|
+
require_relative 'status/unmerged_entry'
|
13
|
+
require_relative 'status/untracked_entry'
|
14
|
+
|
15
|
+
module RubyGit
|
16
|
+
# The working tree status
|
17
|
+
module Status
|
18
|
+
# Parse output of `git status` and return a structured report
|
19
|
+
#
|
20
|
+
# @example
|
21
|
+
# output = `git status -u --porcelain=v2 --renames --branch --show-stash -z`
|
22
|
+
# status = RubyGit::Status.parse(output)
|
23
|
+
# status.branch.name #=> 'main'
|
24
|
+
#
|
25
|
+
# @param status_output [String] the raw output from git status command
|
26
|
+
# @return [RubyGit::Status::Report] a structured representation of git status
|
27
|
+
#
|
28
|
+
# @api public
|
29
|
+
def self.parse(status_output)
|
30
|
+
Parser.parse(status_output)
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
data/lib/ruby_git/version.rb
CHANGED
@@ -0,0 +1,277 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module RubyGit
|
4
|
+
# The working tree is a directory tree consisting of the checked out files that
|
5
|
+
# you are currently working on.
|
6
|
+
#
|
7
|
+
# Create a new Worktree using {.init}, {.clone}, or {.open}.
|
8
|
+
#
|
9
|
+
class Worktree
|
10
|
+
# The root path of the working tree
|
11
|
+
#
|
12
|
+
# @example
|
13
|
+
# worktree_path = '/Users/James/myproject'
|
14
|
+
# worktree = Worktree.open(worktree_path)
|
15
|
+
# worktree.path
|
16
|
+
# => '/Users/James/myproject'
|
17
|
+
#
|
18
|
+
# @return [Pathname] the root path of the worktree
|
19
|
+
#
|
20
|
+
attr_reader :path
|
21
|
+
|
22
|
+
# Create an empty Git repository under the root working tree `path`
|
23
|
+
#
|
24
|
+
# If the repository already exists, it will not be overwritten.
|
25
|
+
#
|
26
|
+
# @see https://git-scm.com/docs/git-init git-init
|
27
|
+
#
|
28
|
+
# @example
|
29
|
+
# worktree = Worktree.init(worktree_path)
|
30
|
+
#
|
31
|
+
# @param [String] worktree_path the root path of a Git working tree
|
32
|
+
#
|
33
|
+
# @raise [RubyGit::Error] if worktree_path is not a directory
|
34
|
+
#
|
35
|
+
# @return [RubyGit::Worktree] the working tree whose root is at `path`
|
36
|
+
#
|
37
|
+
def self.init(worktree_path)
|
38
|
+
raise RubyGit::Error, "Path '#{worktree_path}' not valid." unless File.directory?(worktree_path)
|
39
|
+
|
40
|
+
command = ['init']
|
41
|
+
options = { chdir: worktree_path, out: StringIO.new, err: StringIO.new }
|
42
|
+
RubyGit::CommandLine.run(*command, **options)
|
43
|
+
|
44
|
+
new(worktree_path)
|
45
|
+
end
|
46
|
+
|
47
|
+
# Open an existing Git working tree that contains worktree_path
|
48
|
+
#
|
49
|
+
# @see https://git-scm.com/docs/git-open git-open
|
50
|
+
#
|
51
|
+
# @example
|
52
|
+
# worktree = Worktree.open(worktree_path)
|
53
|
+
#
|
54
|
+
# @param [String] worktree_path the root path of a Git working tree
|
55
|
+
#
|
56
|
+
# @raise [RubyGit::Error] if `worktree_path` does not exist, is not a directory, or is not within
|
57
|
+
# a Git working tree.
|
58
|
+
#
|
59
|
+
# @return [RubyGit::Worktree] the Git working tree that contains `worktree_path`
|
60
|
+
#
|
61
|
+
def self.open(worktree_path)
|
62
|
+
new(worktree_path)
|
63
|
+
end
|
64
|
+
|
65
|
+
# Copy the remote repository and checkout the default branch
|
66
|
+
#
|
67
|
+
# Clones the repository referred to by `repository_url` into a newly created
|
68
|
+
# directory, creates remote-tracking branches for each branch in the cloned repository,
|
69
|
+
# and checks out the default branch in the Git working tree whose root directory is `to_path`.
|
70
|
+
#
|
71
|
+
# @see https://git-scm.com/docs/git-clone git-clone
|
72
|
+
#
|
73
|
+
# @example Using default for Worktree path
|
74
|
+
# FileUtils.pwd
|
75
|
+
# => "/Users/jsmith"
|
76
|
+
# worktree = Worktree.clone('https://github.com/main-branch/ruby_git.git')
|
77
|
+
# worktree.path
|
78
|
+
# => "/Users/jsmith/ruby_git"
|
79
|
+
#
|
80
|
+
# @example Using a specified worktree_path
|
81
|
+
# FileUtils.pwd
|
82
|
+
# => "/Users/jsmith"
|
83
|
+
# worktree_path = '/tmp/project'
|
84
|
+
# worktree = Worktree.clone('https://github.com/main-branch/ruby_git.git', to_path: worktree_path)
|
85
|
+
# worktree.path
|
86
|
+
# => "/tmp/project"
|
87
|
+
#
|
88
|
+
# @param [String] repository_url a reference to a Git repository
|
89
|
+
#
|
90
|
+
# @param [String] to_path where to put the checked out Git working tree once the repository is cloned
|
91
|
+
#
|
92
|
+
# `to_path` will be created if it does not exist. An error is raised if `to_path` exists and
|
93
|
+
# not an empty directory.
|
94
|
+
#
|
95
|
+
# @raise [RubyGit::FailedError] if (1) `repository_url` is not valid or does not point to a valid repository OR
|
96
|
+
# (2) `to_path` is not an empty directory.
|
97
|
+
#
|
98
|
+
# @return [RubyGit::Worktree] the Git working tree checked out from the cloned repository
|
99
|
+
#
|
100
|
+
def self.clone(repository_url, to_path: nil)
|
101
|
+
command = ['clone', '--', repository_url]
|
102
|
+
command << to_path if to_path
|
103
|
+
options = { out: StringIO.new, err: StringIO.new }
|
104
|
+
clone_output = RubyGit::CommandLine.run(*command, **options).stderr
|
105
|
+
new(cloned_to(clone_output))
|
106
|
+
end
|
107
|
+
|
108
|
+
# Get path of the cloned worktree from `git clone` stderr output
|
109
|
+
#
|
110
|
+
# @param clone_output [String] the stderr output of the `git clone` command
|
111
|
+
#
|
112
|
+
# @return [String] the path of the cloned worktree
|
113
|
+
#
|
114
|
+
# @api private
|
115
|
+
def self.cloned_to(clone_output)
|
116
|
+
clone_output.match(/Cloning into ['"](.+)['"]\.\.\./)[1]
|
117
|
+
end
|
118
|
+
|
119
|
+
# Show the working tree and index status
|
120
|
+
#
|
121
|
+
# @example worktree = Worktree.open(worktree_path) worktree.status #=>
|
122
|
+
# #<RubyGit::Status::Report ...>
|
123
|
+
#
|
124
|
+
# @param path_specs [Array<String>] paths to limit the status to
|
125
|
+
# (default is all paths)
|
126
|
+
#
|
127
|
+
# See [git-glossary
|
128
|
+
# pathspec](https://git-scm.com/docs/gitglossary#Documentation/gitglossary.txt-aiddefpathspecapathspec).
|
129
|
+
#
|
130
|
+
# @param untracked_files [:all, :normal, :no] Defines how untracked files will be
|
131
|
+
# handled
|
132
|
+
#
|
133
|
+
# See [git-staus
|
134
|
+
# --untracked-files](https://git-scm.com/docs/git-status#Documentation/git-status.txt---untracked-filesltmodegt).
|
135
|
+
#
|
136
|
+
# @param ignored [:traditional, :matching, :no] Defines how ignored files will be
|
137
|
+
# handled, :no to not include ignored files
|
138
|
+
#
|
139
|
+
# See [git-staus
|
140
|
+
# --ignored](https://git-scm.com/docs/git-status#Documentation/git-status.txt---ignoredltmodegt).
|
141
|
+
#
|
142
|
+
# @param ignore_submodules [:all, :dirty, :untracked, :none] Default is :all
|
143
|
+
#
|
144
|
+
# See [git-staus
|
145
|
+
# --ignore-submodules](https://git-scm.com/docs/git-status#Documentation/git-status.txt---ignore-submodulesltwhengt).
|
146
|
+
#
|
147
|
+
# @return [RubyGit::Status::Report] the status of the working tree
|
148
|
+
#
|
149
|
+
def status(*path_specs, untracked_files: :all, ignored: :no, ignore_submodules: :all)
|
150
|
+
command = %w[status --porcelain=v2 --branch --show-stash --ahead-behind --renames -z]
|
151
|
+
command << "--untracked-files=#{untracked_files}"
|
152
|
+
command << "--ignored=#{ignored}"
|
153
|
+
command << "--ignore-submodules=#{ignore_submodules}"
|
154
|
+
command << '--' unless path_specs.empty?
|
155
|
+
command.concat(path_specs)
|
156
|
+
options = { out: StringIO.new, err: StringIO.new }
|
157
|
+
status_output = run(*command, **options).stdout
|
158
|
+
RubyGit::Status.parse(status_output)
|
159
|
+
end
|
160
|
+
|
161
|
+
# Add changed files to the index to stage for the next commit
|
162
|
+
#
|
163
|
+
# @example
|
164
|
+
# worktree = Worktree.open(worktree_path)
|
165
|
+
# worktree.add('file1.txt', 'file2.txt')
|
166
|
+
# worktree.add('.')
|
167
|
+
# worktree.add(all: true)
|
168
|
+
#
|
169
|
+
# @param pathspecs [Array<String>] paths to add to the index
|
170
|
+
# @param all [Boolean] adds, updates, and removes index entries to match the working tree (entire repo)
|
171
|
+
# @param force [Boolean] add files even if they are ignored
|
172
|
+
# @param refresh [Boolean] only refresh each files stat information in the index
|
173
|
+
# @param update [Boolean] add all updated and deleted files to the index but does not add any files
|
174
|
+
#
|
175
|
+
# @see https://git-scm.com/docs/git-add git-add
|
176
|
+
#
|
177
|
+
# @return [RubyGit::CommandLineResult] the result of the git add command
|
178
|
+
#
|
179
|
+
# @raise [ArgumentError] if any of the options are not valid
|
180
|
+
#
|
181
|
+
def add(*pathspecs, all: false, force: false, refresh: false, update: false) # rubocop:disable Metrics/MethodLength
|
182
|
+
validate_boolean_option(name: :all, value: all)
|
183
|
+
validate_boolean_option(name: :force, value: force)
|
184
|
+
validate_boolean_option(name: :refresh, value: refresh)
|
185
|
+
validate_boolean_option(name: :update, value: update)
|
186
|
+
|
187
|
+
command = %w[add]
|
188
|
+
command << '--all' if all
|
189
|
+
command << '--force' if force
|
190
|
+
command << '--update' if update
|
191
|
+
command << '--refresh' if refresh
|
192
|
+
command << '--' unless pathspecs.empty?
|
193
|
+
command.concat(pathspecs)
|
194
|
+
|
195
|
+
options = { out: StringIO.new, err: StringIO.new }
|
196
|
+
|
197
|
+
run(*command, **options)
|
198
|
+
end
|
199
|
+
|
200
|
+
# Return the repository associated with the worktree
|
201
|
+
#
|
202
|
+
# @example
|
203
|
+
# worktree = Worktree.open(worktree_path)
|
204
|
+
# worktree.repository #=> #<RubyGit::Repository ...>
|
205
|
+
#
|
206
|
+
# @return [RubyGit::Repository] the repository associated with the worktree
|
207
|
+
#
|
208
|
+
def repository
|
209
|
+
@repository ||= begin
|
210
|
+
command = %w[rev-parse --git-dir]
|
211
|
+
options = { chdir: path, chomp: true, out: StringIO.new, err: StringIO.new }
|
212
|
+
# rev-parse path might be relative to the worktree, thus the need to expand it
|
213
|
+
git_dir = File.realpath(RubyGit::CommandLine.run(*command, **options).stdout, path)
|
214
|
+
Repository.new(git_dir)
|
215
|
+
end
|
216
|
+
end
|
217
|
+
|
218
|
+
private
|
219
|
+
|
220
|
+
# Create a Worktree object
|
221
|
+
#
|
222
|
+
# @param worktree_path [String] a path anywhere in the worktree
|
223
|
+
#
|
224
|
+
# @api private
|
225
|
+
#
|
226
|
+
def initialize(worktree_path)
|
227
|
+
raise RubyGit::Error, "Path '#{worktree_path}' not valid." unless File.directory?(worktree_path)
|
228
|
+
|
229
|
+
@path = root_path(worktree_path)
|
230
|
+
RubyGit.logger.debug("Created #{inspect}")
|
231
|
+
end
|
232
|
+
|
233
|
+
# Find the root path of a Git working tree containing `path`
|
234
|
+
#
|
235
|
+
# @raise [RubyGit::FailedError] if the path is not in a Git working tree
|
236
|
+
#
|
237
|
+
# @return [String] the root path of the Git working tree containing `path`
|
238
|
+
#
|
239
|
+
# @api private
|
240
|
+
#
|
241
|
+
def root_path(worktree_path)
|
242
|
+
command = %w[rev-parse --show-toplevel]
|
243
|
+
options = { chdir: worktree_path, chomp: true, out: StringIO.new, err: StringIO.new }
|
244
|
+
File.realpath(RubyGit::CommandLine.run(*command, **options).stdout)
|
245
|
+
end
|
246
|
+
|
247
|
+
# Run a Git command in this worktree
|
248
|
+
#
|
249
|
+
# Passes the repository path and worktree path to RubyGit::CommandLine.run
|
250
|
+
#
|
251
|
+
# @param command [Array<String>] the git command to run
|
252
|
+
# @param options [Hash] options to pass to RubyGit::CommandLine.run
|
253
|
+
#
|
254
|
+
# @return [RubyGit::CommandLineResult] the result of the git command
|
255
|
+
#
|
256
|
+
# @api private
|
257
|
+
#
|
258
|
+
def run(*command, **options)
|
259
|
+
RubyGit::CommandLine.run(*command, repository_path: repository.path, worktree_path: path, **options)
|
260
|
+
end
|
261
|
+
|
262
|
+
# Raise an error if an option is not a Boolean (or optionally nil) value
|
263
|
+
# @param name [String] the name of the option
|
264
|
+
# @param value [Object] the value of the option
|
265
|
+
# @param nullable [Boolean] whether the option can be nil (default is false)
|
266
|
+
# @return [void]
|
267
|
+
# @raise [ArgumentError] if the option is not a Boolean (or optionally nil) value
|
268
|
+
# @api private
|
269
|
+
def validate_boolean_option(name:, value:, nullable: false)
|
270
|
+
return if nullable && value.nil?
|
271
|
+
|
272
|
+
return if [true, false].include?(value)
|
273
|
+
|
274
|
+
raise ArgumentError, "The '#{name}:' option must be a Boolean value but was #{value.inspect}"
|
275
|
+
end
|
276
|
+
end
|
277
|
+
end
|