Module: Git::Parsers::Tag Private
- Defined in:
- lib/git/parsers/tag.rb
Overview
This module is part of a private API. You should avoid using this module if possible, as it may be removed or be changed in the future.
Known limitation: If a tag message contains the field delimiter character (\x1f, ASCII unit separator), it will be preserved correctly since the message is the last field. However, messages are rarely crafted with non-printable control characters.
Parser for git tag command output
Handles parsing of git tag --list and git tag --delete output
into structured data objects.
Design Note: Namespace Organization
This parser creates and returns TagInfo and TagDeleteResult
objects, which live at the top-level Git:: namespace rather than within
Git::Parsers::. This is intentional:
- Parsers are infrastructure - marked
@api private, users shouldn't interact with them directly - Info/Result classes are public API - returned by commands and used throughout the codebase
- Info classes are domain entities - represent core git concepts (tags as data)
- Result classes are operation outcomes - represent command results, not parsing details
Keeping Info/Result classes at Git:: improves discoverability and correctly
reflects their role as public types rather than parser internals.
Constant Summary collapse
- FIELD_DELIMITER =
This constant is part of a private API. You should avoid using this constant if possible, as it may be removed or be changed in the future.
Delimiter for separating fields in git tag --format output Field separator used in custom format output Using the ASCII unit separator (US, 0x1F / "\x1f"), a non-printable character, minimizes the chance of collisions with tag names or messages and remains safe to pass through Process.spawn and shell argument boundaries.
"\x1f"- RECORD_DELIMITER =
This constant is part of a private API. You should avoid using this constant if possible, as it may be removed or be changed in the future.
Delimiter for separating records (tags) in output Using the ASCII record separator (RS, 0x1E / "\x1e") to delimit complete tag records. This allows multi-line messages (which contain newlines) to be parsed correctly since we split by record separator first, then by field delimiter.
"\x1e"- FIELD_COUNT =
This constant is part of a private API. You should avoid using this constant if possible, as it may be removed or be changed in the future.
Number of fields expected in the parsed output
8- FORMAT_STRING =
This constant is part of a private API. You should avoid using this constant if possible, as it may be removed or be changed in the future.
Format string for git tag --format
Fields:
- %(refname:short) - tag name
- %(objectname) - SHA of the tag object (for annotated) or commit (for lightweight)
- %(*objectname) - Dereferenced SHA (commit ID for annotated tags, empty for lightweight)
- %(objecttype) - 'tag' for annotated tags, target object type (commit/tree/blob/etc.) for lightweight tags
- %(taggername) - tagger name (empty for lightweight tags)
- %(taggeremail) - tagger email (empty for lightweight tags)
- %(taggerdate:iso8601-strict) - tagger date in strict ISO 8601 format
- %(contents) - full tag message (can be multi-line)
Each tag record is terminated by the RECORD_DELIMITER to allow multi-line messages.
[ '%(refname:short)', '%(objectname)', '%(*objectname)', '%(objecttype)', '%(taggername)', '%(taggeremail)', '%(taggerdate:iso8601-strict)', '%(contents)' ].join(FIELD_DELIMITER) + RECORD_DELIMITER
- DELETED_TAG_REGEX =
This constant is part of a private API. You should avoid using this constant if possible, as it may be removed or be changed in the future.
Regex to parse successful deletion lines from stdout Matches: Deleted tag 'tagname' (was abc123)
/^Deleted tag '([^']+)'/- ERROR_TAG_REGEX =
This constant is part of a private API. You should avoid using this constant if possible, as it may be removed or be changed in the future.
Regex to parse error messages from stderr Matches: error: tag 'tagname' not found.
/^error: tag '([^']+)'(.*)$/
Class Method Summary collapse
-
.build_delete_result(requested_names, existing_tags, deleted_names, error_map) ⇒ Git::TagDeleteResult
private
Build the TagDeleteResult from parsed data.
-
.build_tag_info(parts) ⇒ Git::TagInfo
private
Build a TagInfo object from parsed parts.
- .build_tag_info_object(parts, oid, target_oid) private
-
.parse_deleted_tags(stdout) ⇒ Array<String>
private
Parse deleted tag names from stdout.
-
.parse_error_messages(stderr) ⇒ Hash<String, String>
private
Parse error messages from stderr into a map.
-
.parse_list(stdout) ⇒ Array<Git::TagInfo>
private
Parse git tag --list output into TagInfo objects.
-
.parse_message(objecttype, message) ⇒ String?
private
Parse message field, returning nil for lightweight tags or empty messages Strips trailing newlines that git adds to %(contents) output.
-
.parse_optional_field(value) ⇒ String?
private
Parse an optional field, returning nil if empty.
-
.parse_tag_record(record, index, all_records) ⇒ Git::TagInfo
private
Parse a single formatted tag record.
- .resolve_oids(objecttype, objectname, dereferenced) private
-
.unexpected_tag_record_error(records, record, index) ⇒ String
private
Generate error message for unexpected tag record format.
Class Method Details
.build_delete_result(requested_names, existing_tags, deleted_names, error_map) ⇒ Git::TagDeleteResult
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Build the TagDeleteResult from parsed data
223 224 225 226 227 228 229 230 231 232 |
# File 'lib/git/parsers/tag.rb', line 223 def build_delete_result(requested_names, , deleted_names, error_map) deleted = deleted_names.filter_map { |name| [name] } not_deleted = (requested_names - deleted_names).map do |name| = error_map[name] || "tag '#{name}' could not be deleted" Git::TagDeleteFailure.new(name: name, error_message: ) end Git::TagDeleteResult.new(deleted: deleted, not_deleted: not_deleted) end |
.build_tag_info(parts) ⇒ Git::TagInfo
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
For annotated tags:
- oid = %(objectname) (the tag object's ID)
- target_oid = %(*objectname) (the dereferenced commit ID)
For lightweight tags:
- oid = nil (lightweight tags are not objects)
- target_oid = %(objectname) (the commit ID)
Build a TagInfo object from parsed parts
148 149 150 151 |
# File 'lib/git/parsers/tag.rb', line 148 def build_tag_info(parts) oid, target_oid = resolve_oids(parts[3], parts[1], parts[2]) build_tag_info_object(parts, oid, target_oid) end |
.build_tag_info_object(parts, oid, target_oid)
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
157 158 159 160 161 162 163 |
# File 'lib/git/parsers/tag.rb', line 157 def build_tag_info_object(parts, oid, target_oid) Git::TagInfo.new( name: parts[0], oid: oid, target_oid: target_oid, objecttype: parts[3], tagger_name: parse_optional_field(parts[4]), tagger_email: parse_optional_field(parts[5]), tagger_date: parse_optional_field(parts[6]), message: (parts[3], parts[7]) ) end |
.parse_deleted_tags(stdout) ⇒ Array<String>
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Parse deleted tag names from stdout
195 196 197 |
# File 'lib/git/parsers/tag.rb', line 195 def (stdout) stdout.scan(DELETED_TAG_REGEX).flatten end |
.parse_error_messages(stderr) ⇒ Hash<String, String>
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Parse error messages from stderr into a map
208 209 210 211 212 213 |
# File 'lib/git/parsers/tag.rb', line 208 def (stderr) stderr.each_line.with_object({}) do |line, hash| match = line.match(ERROR_TAG_REGEX) hash[match[1]] = line.strip if match end end |
.parse_list(stdout) ⇒ Array<Git::TagInfo>
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Parse git tag --list output into TagInfo objects
101 102 103 104 105 106 107 |
# File 'lib/git/parsers/tag.rb', line 101 def parse_list(stdout) # Split by record separator # Each record may have a leading newline from the previous record's %(contents) output # Use lstrip to remove leading whitespace (which includes the newline) from each record records = stdout.split(RECORD_DELIMITER).map(&:lstrip).reject(&:empty?) records.map.with_index { |record, index| parse_tag_record(record, index, records) } end |
.parse_message(objecttype, message) ⇒ String?
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Parse message field, returning nil for lightweight tags or empty messages Strips trailing newlines that git adds to %(contents) output
181 182 183 184 |
# File 'lib/git/parsers/tag.rb', line 181 def (objecttype, ) stripped = .chomp objecttype == 'tag' && !stripped.empty? ? stripped : nil end |
.parse_optional_field(value) ⇒ String?
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Parse an optional field, returning nil if empty
170 171 172 |
# File 'lib/git/parsers/tag.rb', line 170 def parse_optional_field(value) value.empty? ? nil : value end |
.parse_tag_record(record, index, all_records) ⇒ Git::TagInfo
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Parse a single formatted tag record
The record format is:
name
For lightweight tags, Git emits empty strings for the tagger fields and message; these are converted to nil by #parse_optional_field and #parse_message.
125 126 127 128 129 130 131 132 133 |
# File 'lib/git/parsers/tag.rb', line 125 def parse_tag_record(record, index, all_records) parts = record.split(FIELD_DELIMITER, FIELD_COUNT) unless parts.length == FIELD_COUNT raise Git::UnexpectedResultError, unexpected_tag_record_error(all_records, record, index) end build_tag_info(parts) end |
.resolve_oids(objecttype, objectname, dereferenced)
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
153 154 155 |
# File 'lib/git/parsers/tag.rb', line 153 def resolve_oids(objecttype, objectname, dereferenced) objecttype == 'tag' ? [objectname, dereferenced] : [nil, objectname] end |
.unexpected_tag_record_error(records, record, index) ⇒ String
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Generate error message for unexpected tag record format
241 242 243 244 245 246 247 248 249 250 251 252 253 254 |
# File 'lib/git/parsers/tag.rb', line 241 def unexpected_tag_record_error(records, record, index) format_str = FORMAT_STRING.gsub(FIELD_DELIMITER, '<FS>').gsub(RECORD_DELIMITER, '<RS>') <<~ERROR Unexpected record in output from `git tag --list --format=#{format_str}`, at index #{index} Expected #{FIELD_COUNT} fields separated by '\\x1f' (unit separator), got #{record.split(FIELD_DELIMITER, -1).length} Full output: #{records.join("\n ")} Record at index #{index}: "#{record}" ERROR end |