ruby_git 0.2.0 → 0.3.1

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 (55) hide show
  1. checksums.yaml +4 -4
  2. data/.commitlintrc.yml +16 -0
  3. data/.github/CODEOWNERS +4 -0
  4. data/.github/workflows/continuous_integration.yml +87 -0
  5. data/.github/workflows/enforce_conventional_commits.yml +21 -0
  6. data/.github/workflows/experimental_ruby_builds.yml +65 -0
  7. data/.gitignore +3 -0
  8. data/.husky/commit-msg +1 -0
  9. data/.markdownlint.yml +25 -0
  10. data/.rubocop.yml +13 -15
  11. data/.yardopts +6 -1
  12. data/CHANGELOG.md +58 -0
  13. data/CONTRIBUTING.md +5 -5
  14. data/{LICENSE.md → LICENSE.txt} +1 -1
  15. data/PLAN.md +67 -0
  16. data/README.md +58 -6
  17. data/RELEASING.md +5 -54
  18. data/Rakefile +31 -38
  19. data/RubyGit Class Diagram.svg +1 -0
  20. data/bin/command-line-test +189 -0
  21. data/bin/console +2 -0
  22. data/bin/setup +13 -2
  23. data/lib/ruby_git/command_line/options.rb +61 -0
  24. data/lib/ruby_git/command_line/result.rb +155 -0
  25. data/lib/ruby_git/command_line/runner.rb +296 -0
  26. data/lib/ruby_git/command_line.rb +95 -0
  27. data/lib/ruby_git/encoding_normalizer.rb +49 -0
  28. data/lib/ruby_git/errors.rb +169 -0
  29. data/lib/ruby_git/repository.rb +33 -0
  30. data/lib/ruby_git/status/branch.rb +92 -0
  31. data/lib/ruby_git/status/entry.rb +162 -0
  32. data/lib/ruby_git/status/ignored_entry.rb +44 -0
  33. data/lib/ruby_git/status/ordinary_entry.rb +207 -0
  34. data/lib/ruby_git/status/parser.rb +203 -0
  35. data/lib/ruby_git/status/renamed_entry.rb +257 -0
  36. data/lib/ruby_git/status/report.rb +143 -0
  37. data/lib/ruby_git/status/stash.rb +33 -0
  38. data/lib/ruby_git/status/submodule_status.rb +85 -0
  39. data/lib/ruby_git/status/unmerged_entry.rb +248 -0
  40. data/lib/ruby_git/status/untracked_entry.rb +52 -0
  41. data/lib/ruby_git/status.rb +33 -0
  42. data/lib/ruby_git/version.rb +1 -1
  43. data/lib/ruby_git/worktree.rb +175 -33
  44. data/lib/ruby_git.rb +91 -28
  45. data/package.json +11 -0
  46. data/ruby_git.gemspec +33 -23
  47. metadata +146 -50
  48. data/.travis.yml +0 -26
  49. data/CODEOWNERS +0 -3
  50. data/MAINTAINERS.md +0 -8
  51. data/lib/ruby_git/error.rb +0 -8
  52. data/lib/ruby_git/file_helpers.rb +0 -42
  53. data/lib/ruby_git/git_binary.rb +0 -106
  54. /data/{ISSUE_TEMPLATE.md → .github/ISSUE_TEMPLATE.md} +0 -0
  55. /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
@@ -3,5 +3,5 @@
3
3
  module RubyGit
4
4
  # The ruby_git gem version
5
5
  #
6
- VERSION = '0.2.0'
6
+ VERSION = '0.3.1'
7
7
  end
@@ -1,15 +1,13 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'open3'
4
-
5
3
  module RubyGit
6
- # The Worktree is a directory tree consisting of the checked out files that
4
+ # The working tree is a directory tree consisting of the checked out files that
7
5
  # you are currently working on.
8
6
  #
9
7
  # Create a new Worktree using {.init}, {.clone}, or {.open}.
10
8
  #
11
9
  class Worktree
12
- # The root path of the worktree
10
+ # The root path of the working tree
13
11
  #
14
12
  # @example
15
13
  # worktree_path = '/Users/James/myproject'
@@ -21,7 +19,7 @@ module RubyGit
21
19
  #
22
20
  attr_reader :path
23
21
 
24
- # Create an empty Git repository under the root worktree `path`
22
+ # Create an empty Git repository under the root working tree `path`
25
23
  #
26
24
  # If the repository already exists, it will not be overwritten.
27
25
  #
@@ -30,34 +28,35 @@ module RubyGit
30
28
  # @example
31
29
  # worktree = Worktree.init(worktree_path)
32
30
  #
33
- # @param [String] worktree_path the root path of a worktree
31
+ # @param [String] worktree_path the root path of a Git working tree
34
32
  #
35
33
  # @raise [RubyGit::Error] if worktree_path is not a directory
36
34
  #
37
- # @return [RubyGit::Worktree] the worktree whose root is at `path`
35
+ # @return [RubyGit::Worktree] the working tree whose root is at `path`
38
36
  #
39
37
  def self.init(worktree_path)
40
38
  raise RubyGit::Error, "Path '#{worktree_path}' not valid." unless File.directory?(worktree_path)
41
39
 
42
- command = [RubyGit.git.path.to_s, 'init']
43
- _out, err, status = Open3.capture3(*command, chdir: worktree_path)
44
- raise RubyGit::Error, err unless status.success?
40
+ command = ['init']
41
+ options = { chdir: worktree_path, out: StringIO.new, err: StringIO.new }
42
+ RubyGit::CommandLine.run(*command, **options)
45
43
 
46
- Worktree.new(worktree_path)
44
+ new(worktree_path)
47
45
  end
48
46
 
49
- # Open an existing Git worktree that contains worktree_path
47
+ # Open an existing Git working tree that contains worktree_path
50
48
  #
51
49
  # @see https://git-scm.com/docs/git-open git-open
52
50
  #
53
51
  # @example
54
52
  # worktree = Worktree.open(worktree_path)
55
53
  #
56
- # @param [String] worktree_path the root path of a worktree
54
+ # @param [String] worktree_path the root path of a Git working tree
57
55
  #
58
- # @raise [RubyGit::Error] if `worktree_path` does not exist, is not a directory, or is not within a Git worktree.
56
+ # @raise [RubyGit::Error] if `worktree_path` does not exist, is not a directory, or is not within
57
+ # a Git working tree.
59
58
  #
60
- # @return [RubyGit::Worktree] the worktree that contains `worktree_path`
59
+ # @return [RubyGit::Worktree] the Git working tree that contains `worktree_path`
61
60
  #
62
61
  def self.open(worktree_path)
63
62
  new(worktree_path)
@@ -67,7 +66,7 @@ module RubyGit
67
66
  #
68
67
  # Clones the repository referred to by `repository_url` into a newly created
69
68
  # directory, creates remote-tracking branches for each branch in the cloned repository,
70
- # and checks out the default branch in the worktree whose root directory is `to_path`.
69
+ # and checks out the default branch in the Git working tree whose root directory is `to_path`.
71
70
  #
72
71
  # @see https://git-scm.com/docs/git-clone git-clone
73
72
  #
@@ -76,7 +75,7 @@ module RubyGit
76
75
  # => "/Users/jsmith"
77
76
  # worktree = Worktree.clone('https://github.com/main-branch/ruby_git.git')
78
77
  # worktree.path
79
- # => "/Users/jsmith/ruby_git"
78
+ # => "/Users/jsmith/ruby_git"
80
79
  #
81
80
  # @example Using a specified worktree_path
82
81
  # FileUtils.pwd
@@ -84,52 +83,195 @@ module RubyGit
84
83
  # worktree_path = '/tmp/project'
85
84
  # worktree = Worktree.clone('https://github.com/main-branch/ruby_git.git', to_path: worktree_path)
86
85
  # worktree.path
87
- # => "/tmp/project"
86
+ # => "/tmp/project"
88
87
  #
89
88
  # @param [String] repository_url a reference to a Git repository
90
89
  #
91
- # @param [String] to_path where to put the checked out worktree once the repository is cloned
90
+ # @param [String] to_path where to put the checked out Git working tree once the repository is cloned
92
91
  #
93
92
  # `to_path` will be created if it does not exist. An error is raised if `to_path` exists and
94
93
  # not an empty directory.
95
94
  #
96
- # @raise [RubyGit::Error] if (1) `repository_url` is not valid or does not point to a valid repository OR
95
+ # @raise [RubyGit::FailedError] if (1) `repository_url` is not valid or does not point to a valid repository OR
97
96
  # (2) `to_path` is not an empty directory.
98
97
  #
99
- # @return [RubyGit::Worktree] the worktree checked out from the cloned repository
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
100
113
  #
101
- def self.clone(repository_url, to_path: '')
102
- command = [RubyGit.git.path.to_s, 'clone', '--', repository_url, to_path]
103
- _out, err, status = Open3.capture3(*command)
104
- raise RubyGit::Error, err unless status.success?
114
+ # @api private
115
+ def self.cloned_to(clone_output)
116
+ clone_output.match(/Cloning into ['"](.+)['"]\.\.\./)[1]
117
+ end
105
118
 
106
- new(to_path)
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
107
216
  end
108
217
 
109
218
  private
110
219
 
111
220
  # Create a Worktree object
221
+ #
222
+ # @param worktree_path [String] a path anywhere in the worktree
223
+ #
112
224
  # @api private
225
+ #
113
226
  def initialize(worktree_path)
114
227
  raise RubyGit::Error, "Path '#{worktree_path}' not valid." unless File.directory?(worktree_path)
115
228
 
116
229
  @path = root_path(worktree_path)
230
+ RubyGit.logger.debug("Created #{inspect}")
117
231
  end
118
232
 
119
- # Find the root path of a worktree containing `path`
233
+ # Find the root path of a Git working tree containing `path`
120
234
  #
121
- # @raise [RubyGit::Error] if the path is not in a worktree
235
+ # @raise [RubyGit::FailedError] if the path is not in a Git working tree
122
236
  #
123
- # @return [String] the root path of the worktree containing `path`
237
+ # @return [String] the root path of the Git working tree containing `path`
124
238
  #
125
239
  # @api private
126
240
  #
127
241
  def root_path(worktree_path)
128
- command = [RubyGit.git.path.to_s, 'rev-parse', '--show-toplevel']
129
- out, err, status = Open3.capture3(*command, chdir: worktree_path)
130
- raise RubyGit::Error, err unless status.success?
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)
131
273
 
132
- out.chomp
274
+ raise ArgumentError, "The '#{name}:' option must be a Boolean value but was #{value.inspect}"
133
275
  end
134
276
  end
135
277
  end