diagram 0.3.0 → 0.3.3

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 (47) hide show
  1. checksums.yaml +4 -4
  2. data/lib/diagram.rb +3 -0
  3. data/lib/diagrams/base.rb +8 -6
  4. data/lib/diagrams/class_diagram.rb +0 -4
  5. data/lib/diagrams/elements/class_entity.rb +0 -4
  6. data/lib/diagrams/elements/edge.rb +0 -4
  7. data/lib/diagrams/elements/erd_attribute.rb +28 -0
  8. data/lib/diagrams/elements/erd_entity.rb +23 -0
  9. data/lib/diagrams/elements/erd_relationship.rb +40 -0
  10. data/lib/diagrams/elements/event.rb +0 -4
  11. data/lib/diagrams/elements/gantt_section.rb +23 -0
  12. data/lib/diagrams/elements/git_branch.rb +27 -0
  13. data/lib/diagrams/elements/git_commit.rb +36 -0
  14. data/lib/diagrams/elements/node.rb +0 -8
  15. data/lib/diagrams/elements/relationship.rb +0 -4
  16. data/lib/diagrams/elements/slice.rb +0 -4
  17. data/lib/diagrams/elements/state.rb +0 -4
  18. data/lib/diagrams/elements/task.rb +20 -12
  19. data/lib/diagrams/elements/timeline_event.rb +21 -0
  20. data/lib/diagrams/elements/timeline_period.rb +23 -0
  21. data/lib/diagrams/elements/timeline_section.rb +23 -0
  22. data/lib/diagrams/elements/transition.rb +0 -4
  23. data/lib/diagrams/elements.rb +12 -0
  24. data/lib/diagrams/er_diagram.rb +143 -0
  25. data/lib/diagrams/flowchart_diagram.rb +0 -6
  26. data/lib/diagrams/gantt_diagram.rb +118 -35
  27. data/lib/diagrams/gitgraph_diagram.rb +345 -0
  28. data/lib/diagrams/pie_diagram.rb +0 -3
  29. data/lib/diagrams/state_diagram.rb +0 -5
  30. data/lib/diagrams/timeline_diagram.rb +159 -0
  31. data/lib/diagrams/version.rb +1 -1
  32. data/lib/diagrams.rb +13 -1
  33. data/sig/diagrams/elements/erd_attribute.rbs +20 -0
  34. data/sig/diagrams/elements/erd_entity.rbs +20 -0
  35. data/sig/diagrams/elements/erd_relationship.rbs +25 -0
  36. data/sig/diagrams/elements/gantt_section.rbs +18 -0
  37. data/sig/diagrams/elements/git_branch.rbs +19 -0
  38. data/sig/diagrams/elements/git_commit.rbs +23 -0
  39. data/sig/diagrams/elements/task.rbs +13 -5
  40. data/sig/diagrams/elements/timeline_event.rbs +17 -0
  41. data/sig/diagrams/elements/timeline_period.rbs +18 -0
  42. data/sig/diagrams/elements/timeline_section.rbs +18 -0
  43. data/sig/diagrams/er_diagram.rbs +30 -0
  44. data/sig/diagrams/gantt_diagram.rbs +28 -10
  45. data/sig/diagrams/gitgraph_diagram.rbs +35 -0
  46. data/sig/diagrams/timeline_diagram.rbs +14 -0
  47. metadata +28 -2
@@ -1,87 +1,156 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/task'
5
-
6
3
  module Diagrams
7
- # Represents a Gantt Chart diagram consisting of tasks over time.
4
+ # Represents a Gantt Chart diagram consisting of tasks over time, grouped into sections.
8
5
  class GanttDiagram < Base
9
- attr_reader :title, :tasks
6
+ DEFAULT_SECTION_TITLE = 'Default Section'
7
+
8
+ attr_reader :title, :sections
10
9
 
11
10
  # Initializes a new GanttDiagram.
12
11
  #
13
12
  # @param title [String] The title of the Gantt chart.
14
- # @param tasks [Array<Element::Task>] An array of task objects.
13
+ # @param sections [Array<Element::GanttSection>] An array of section objects (containing tasks).
15
14
  # @param version [String, Integer, nil] User-defined version identifier.
16
- def initialize(title: '', tasks: [], version: 1)
15
+ def initialize(title: '', sections: [], version: 1)
17
16
  super(version:)
18
17
  @title = title.is_a?(String) ? title : ''
19
- @tasks = tasks.is_a?(Array) ? tasks : []
18
+ @sections = sections.is_a?(Array) ? sections : []
19
+ ensure_default_section if @sections.empty?
20
20
  validate_elements!
21
21
  update_checksum!
22
22
  end
23
23
 
24
- # Adds a task to the diagram.
24
+ # Adds a new section to the diagram.
25
+ # Subsequent tasks will be added to this section.
26
+ #
27
+ # @param section_title [String] The title of the section.
28
+ # @raise [ArgumentError] if a section with the same title already exists.
29
+ # @return [Elements::GanttSection] The newly added section.
30
+ def add_section(section_title)
31
+ clean_title = section_title.strip
32
+ raise ArgumentError, "Section title '#{clean_title}' cannot be empty" if clean_title.empty?
33
+ raise ArgumentError, "Section with title '#{clean_title}' already exists" if find_section(clean_title)
34
+
35
+ # Remove default section if it's empty and we're adding a real one
36
+ if @sections.size == 1 && @sections.first.title == DEFAULT_SECTION_TITLE && @sections.first.tasks.empty? # Check tasks for GanttSection
37
+ @sections.clear
38
+ end
39
+
40
+ # Use GanttSection
41
+ new_section = Elements::GanttSection.new(title: clean_title, tasks: [])
42
+ @sections << new_section
43
+ update_checksum!
44
+ new_section
45
+ end
46
+
47
+ # Adds a task to the current (last) section of the diagram.
25
48
  #
26
- # @param task [Element::Task] The task object to add.
27
- # @raise [ArgumentError] if a task with the same ID already exists.
28
- # @return [Element::Task] The added task.
29
- def add_task(task)
30
- raise ArgumentError, 'Task must be a Diagrams::Elements::Task' unless task.is_a?(Diagrams::Elements::Task)
31
- raise ArgumentError, "Task with ID '#{task.id}' already exists" if find_task(task.id)
32
-
33
- @tasks << task
49
+ # @param id [String] Unique ID for the task (used for dependencies).
50
+ # @param label [String] Display name/label for the task.
51
+ # @param status [Symbol, nil] Status (:done, :active, :crit). nil implies default/future.
52
+ # @param start [String] Start date, task ID (e.g., 'task1'), or dependency string ('after taskX').
53
+ # @param duration [String] Duration string (e.g., '7d', '2w').
54
+ # @raise [ArgumentError] if required fields are missing or a task with the same ID exists.
55
+ # @raise [StandardError] if no sections exist.
56
+ # @return [Elements::Task] The added task.
57
+ def add_task(id:, label:, start:, duration:, status: nil)
58
+ raise ArgumentError, 'Task ID cannot be empty' if id.nil? || id.strip.empty?
59
+ raise ArgumentError, "Task with ID '#{id}' already exists" if find_task(id)
60
+
61
+ new_task = Elements::Task.new(
62
+ id:,
63
+ label:,
64
+ status:,
65
+ start:,
66
+ duration:
67
+ )
68
+
69
+ current_section = @sections.last
70
+ raise StandardError, 'Cannot add task: No section available.' unless current_section
71
+
72
+ # Add task to the current section's 'tasks' array
73
+ updated_tasks = current_section.tasks + [new_task]
74
+ updated_section = Elements::GanttSection.new(title: current_section.title, tasks: updated_tasks)
75
+
76
+ # Update the section in the main array
77
+ current_section_index = @sections.index { |s| s.title == current_section.title }
78
+ unless current_section_index
79
+ raise StandardError,
80
+ "Could not find index for current section '#{current_section.title}'"
81
+ end
82
+
83
+ @sections[current_section_index] = updated_section
84
+
34
85
  update_checksum!
35
- task
86
+ new_task
36
87
  end
37
88
 
38
- # Finds a task by its ID.
89
+ # Finds a task by its ID across all sections.
39
90
  #
40
91
  # @param task_id [String] The ID of the task to find.
41
92
  # @return [Element::Task, nil] The found task or nil.
42
93
  def find_task(task_id)
43
- @tasks.find { |t| t.id == task_id }
94
+ all_tasks.find { |t| t.id == task_id }
95
+ end
96
+
97
+ # Finds a section by its title.
98
+ # @param section_title [String] The title of the section.
99
+ # @return [Elements::GanttSection, nil] The found section or nil.
100
+ def find_section(section_title)
101
+ @sections.find { |s| s.title == section_title }
44
102
  end
45
103
 
46
104
  # Returns the specific content of the Gantt diagram as a hash.
47
- # Called by `Diagrams::Base#to_h`.
48
105
  #
49
106
  # @return [Hash{Symbol => String | Array<Hash>}]
50
107
  def to_h_content
51
108
  {
52
109
  title: @title,
53
- tasks: @tasks.map(&:to_h)
110
+ # Serialize sections, renaming 'periods' back to 'tasks' for clarity
111
+ sections: @sections.map(&:to_h) # Use GanttSection's to_h directly
54
112
  }
55
113
  end
56
114
 
57
115
  # Returns a hash mapping element types to their collections for diffing.
58
- # @see Diagrams::Base#identifiable_elements
116
+ #
59
117
  # @return [Hash{Symbol => Array<Diagrams::Elements::Task>}]
60
118
  def identifiable_elements
61
119
  {
62
- tasks: @tasks
120
+ # Diffing based on tasks directly might be more useful than sections here
121
+ tasks: all_tasks
122
+ # sections: @sections # Could also diff sections if needed
63
123
  }
64
124
  end
65
125
 
66
126
  # Class method to create a GanttDiagram from a hash.
67
- # Used by the deserialization factory in `Diagrams::Base`.
68
127
  #
69
- # @param data_hash [Hash] Hash containing `:title` and `:tasks` array.
128
+ # @param data_hash [Hash] Hash containing `:title` and `:sections` array.
70
129
  # @param version [String, Integer, nil] Diagram version.
71
130
  # @param checksum [String, nil] Expected checksum (optional, for verification).
72
131
  # @return [GanttDiagram] The instantiated diagram.
73
132
  def self.from_h(data_hash, version:, checksum:)
74
133
  title = data_hash[:title] || data_hash['title'] || ''
75
- tasks_data = data_hash[:tasks] || data_hash['tasks'] || []
76
-
77
- tasks = tasks_data.map { |task_h| Diagrams::Elements::Task.new(task_h.transform_keys(&:to_sym)) }
134
+ sections_data = data_hash[:sections] || data_hash['sections'] || []
135
+
136
+ sections = sections_data.map do |section_h|
137
+ section_data = section_h.transform_keys(&:to_sym)
138
+ tasks_data = section_data[:tasks] || [] # Expect 'tasks' key in hash
139
+ # Map task data to Task objects
140
+ tasks = tasks_data.map do |task_h|
141
+ task_data = task_h.transform_keys(&:to_sym)
142
+ # Convert status back to symbol if it's a string and present
143
+ task_data[:status] = task_data[:status].to_sym if task_data[:status].is_a?(String)
144
+ Elements::Task.new(task_data)
145
+ end
146
+ Elements::GanttSection.new(title: section_data[:title], tasks: tasks)
147
+ end
78
148
 
79
- diagram = new(title:, tasks:, version:)
149
+ diagram = new(title:, sections:, version:)
80
150
 
81
- # Optional: Verify checksum if provided
151
+ # Optional: Verify checksum
82
152
  if checksum && diagram.checksum != checksum
83
153
  warn "Checksum mismatch for loaded GanttDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
84
- # Or raise an error: raise "Checksum mismatch..."
85
154
  end
86
155
 
87
156
  diagram
@@ -89,14 +158,28 @@ module Diagrams
89
158
 
90
159
  private
91
160
 
161
+ # Helper to get all tasks from all sections.
162
+ def all_tasks
163
+ @sections.flat_map(&:tasks)
164
+ end
165
+
166
+ # Ensures a default section exists if the sections array is empty.
167
+ def ensure_default_section
168
+ return if @sections.any? { |s| s.title == DEFAULT_SECTION_TITLE }
169
+
170
+ @sections << Elements::GanttSection.new(title: DEFAULT_SECTION_TITLE, tasks: [])
171
+ end
172
+
92
173
  # Validates the consistency of tasks during initialization.
93
174
  def validate_elements!
94
- task_ids = @tasks.map(&:id)
95
- return if task_ids.uniq.size == @tasks.size
175
+ task_ids = all_tasks.map(&:id)
176
+ return if task_ids.uniq.size == all_tasks.size
96
177
 
97
178
  raise ArgumentError, 'Duplicate task IDs found'
98
-
99
179
  # Add more validation if needed (e.g., date formats, dependencies)
100
180
  end
181
+
182
+ # Protected method access
183
+ protected :update_checksum!
101
184
  end
102
185
  end
@@ -0,0 +1,345 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'digest' # For generating default commit IDs if needed
4
+
5
+ module Diagrams
6
+ # Represents a Gitgraph diagram, tracking commits, branches, and their relationships.
7
+ class GitgraphDiagram < Base
8
+ attr_reader :commits, :branches, :commit_order, :current_branch_name
9
+
10
+ # Initializes a new GitgraphDiagram.
11
+ # Starts with a 'master' branch by default.
12
+ #
13
+ # @param version [String, Integer, nil] User-defined version identifier.
14
+ def initialize(version: 1)
15
+ super
16
+ @commits = {} # Hash { commit_id => GitCommit }
17
+ @branches = {} # Hash { branch_name => GitBranch }
18
+ @commit_order = [] # Array<String> - IDs of commits in order of creation/operation
19
+ @current_branch_name = 'master'
20
+
21
+ # Initialize main branch conceptually. Its start/head commit will be set by the first commit.
22
+ # We need a placeholder start_commit_id; using a special value or handling nil in GitBranch might be better.
23
+ # For now, let's use a placeholder that signifies it's the root.
24
+ # A better approach might be to create the branch *during* the first commit. Let's refine this.
25
+ # --> Refinement: Don't create the branch object here. Create it during the first 'commit' or 'branch' operation.
26
+ # Initialize @current_branch_name = 'master' conceptually.
27
+
28
+ update_checksum! # Initial checksum for an empty graph
29
+ end
30
+
31
+ # --- Git Operations ---
32
+
33
+ # Adds a commit to the current branch.
34
+ # Handles the creation of the initial 'master' branch on the first commit.
35
+ #
36
+ # @param id [String, nil] Optional custom ID for the commit. Auto-generated if nil.
37
+ # @param message [String, nil] Optional commit message.
38
+ # @param tag [String, nil] Optional tag for the commit.
39
+ # @param type [Symbol] Type of the commit (:NORMAL, :REVERSE, :HIGHLIGHT).
40
+ # @raise [ArgumentError] if a commit with the given ID already exists.
41
+ # @return [Elements::GitCommit] The created commit object.
42
+ def commit(id: nil, message: nil, tag: nil, type: :NORMAL)
43
+ parent_id = current_head_commit_id
44
+ parent_ids = parent_id ? [parent_id] : []
45
+
46
+ commit_id = id || generate_commit_id(parent_ids, message)
47
+ raise ArgumentError, "Commit with ID '#{commit_id}' already exists" if @commits.key?(commit_id)
48
+
49
+ # Handle first commit: create the master branch
50
+ if @commits.empty? && @current_branch_name == 'master' && !@branches.key?('master')
51
+ # The first commit *is* the starting point of the master branch
52
+ master_branch = Elements::GitBranch.new(name: 'master', start_commit_id: commit_id, head_commit_id: commit_id)
53
+ @branches['master'] = master_branch
54
+ elsif !@branches.key?(@current_branch_name)
55
+ # This case shouldn't typically happen if branch/checkout is used correctly,
56
+ # but defensively handle committing to a non-existent branch (other than initial master).
57
+ raise ArgumentError, "Cannot commit: Branch '#{@current_branch_name}' does not exist."
58
+ end
59
+
60
+ new_commit = Elements::GitCommit.new(
61
+ id: commit_id,
62
+ parent_ids:,
63
+ branch_name: @current_branch_name,
64
+ message:,
65
+ tag:,
66
+ type:
67
+ )
68
+
69
+ @commits[commit_id] = new_commit
70
+ @commit_order << commit_id
71
+
72
+ # Update the head of the current branch
73
+ current_branch = @branches[@current_branch_name]
74
+ current_branch.attributes[:head_commit_id] = commit_id # Update using Dry::Struct's way if needed, direct assign might work
75
+
76
+ update_checksum!
77
+ new_commit
78
+ end
79
+
80
+ # Creates a new branch pointing to a specific commit (or the current head)
81
+ # and switches the current context to the new branch.
82
+ #
83
+ # @param name [String] The name for the new branch.
84
+ # @param start_commit_id [String, nil] Optional ID of the commit where the branch should start.
85
+ # Defaults to the head commit of the current branch.
86
+ # @raise [ArgumentError] if the branch name already exists or if trying to branch before any commits exist.
87
+ # @raise [ArgumentError] if a specified `start_commit_id` does not exist.
88
+ # @return [Elements::GitBranch] The created branch object.
89
+ def branch(name:, start_commit_id: nil)
90
+ raise ArgumentError, "Branch name '#{name}' already exists" if @branches.key?(name)
91
+
92
+ effective_start_commit_id = start_commit_id || current_head_commit_id
93
+
94
+ # Ensure there's a commit to branch from
95
+ raise ArgumentError, 'Cannot create a branch before the first commit' unless effective_start_commit_id
96
+
97
+ unless @commits.key?(effective_start_commit_id)
98
+ raise ArgumentError,
99
+ "Start commit ID '#{effective_start_commit_id}' does not exist"
100
+ end
101
+
102
+ new_branch = Elements::GitBranch.new(
103
+ name:,
104
+ # The new branch initially points to the commit it was created from
105
+ start_commit_id: effective_start_commit_id,
106
+ head_commit_id: effective_start_commit_id
107
+ )
108
+
109
+ @branches[name] = new_branch
110
+ @current_branch_name = name # Switch to the new branch
111
+
112
+ update_checksum!
113
+ new_branch
114
+ end
115
+
116
+ # Switches the current context to an existing branch.
117
+ #
118
+ # @param name [String] The name of the branch to switch to.
119
+ # @raise [ArgumentError] if the branch name does not exist.
120
+ # @return [String] The name of the branch checked out.
121
+ def checkout(name:)
122
+ raise ArgumentError, "Branch '#{name}' does not exist. Cannot checkout." unless @branches.key?(name)
123
+
124
+ @current_branch_name = name
125
+ # NOTE: Checkout does not change the diagram structure itself (commits/branches),
126
+ # so we do NOT update the checksum here.
127
+ name
128
+ end
129
+
130
+ # Merges the head of a specified branch into the current branch.
131
+ # Creates a merge commit on the current branch.
132
+ #
133
+ # @param from_branch_name [String] The name of the branch to merge from.
134
+ # @param id [String, nil] Optional custom ID for the merge commit. Auto-generated if nil.
135
+ # @param tag [String, nil] Optional tag for the merge commit.
136
+ # @param type [Symbol] Type of the merge commit (defaults to :MERGE, can be overridden e.g., :REVERSE).
137
+ # @raise [ArgumentError] if `from_branch_name` does not exist, is the same as the current branch,
138
+ # or if either branch has no commits.
139
+ # @raise [ArgumentError] if a commit with the given ID already exists.
140
+ # @return [Elements::GitCommit] The created merge commit object.
141
+ def merge(from_branch_name:, id: nil, tag: nil, type: :MERGE)
142
+ if from_branch_name == @current_branch_name
143
+ raise ArgumentError,
144
+ "Cannot merge branch '#{from_branch_name}' into itself"
145
+ end
146
+ unless @branches.key?(from_branch_name)
147
+ raise ArgumentError,
148
+ "Branch '#{from_branch_name}' does not exist. Cannot merge."
149
+ end
150
+ unless @branches.key?(@current_branch_name)
151
+ raise ArgumentError, "Current branch '#{@current_branch_name}' does not exist. Cannot merge."
152
+ end
153
+
154
+ target_branch = @branches[@current_branch_name]
155
+ source_branch = @branches[from_branch_name]
156
+
157
+ target_head_id = target_branch.head_commit_id
158
+ source_head_id = source_branch.head_commit_id
159
+
160
+ unless target_head_id
161
+ raise ArgumentError,
162
+ "Current branch '#{@current_branch_name}' has no commits to merge into."
163
+ end
164
+ raise ArgumentError, "Source branch '#{from_branch_name}' has no commits to merge from." unless source_head_id
165
+
166
+ # Merge commit parents are the heads of the two branches being merged
167
+ parent_ids = [target_head_id, source_head_id].sort # Sort for consistent checksumming/comparison
168
+
169
+ merge_commit_id = id || generate_commit_id(parent_ids,
170
+ "Merge branch '#{from_branch_name}' into #{@current_branch_name}")
171
+ raise ArgumentError, "Commit with ID '#{merge_commit_id}' already exists" if @commits.key?(merge_commit_id)
172
+
173
+ merge_commit = Elements::GitCommit.new(
174
+ id: merge_commit_id,
175
+ parent_ids:,
176
+ branch_name: @current_branch_name, # Merge commit belongs to the target branch
177
+ message: "Merge branch '#{from_branch_name}' into #{@current_branch_name}", # Default message
178
+ tag:,
179
+ type: # Use provided type, default :MERGE
180
+ )
181
+
182
+ @commits[merge_commit_id] = merge_commit
183
+ @commit_order << merge_commit_id
184
+
185
+ # Update the head of the current (target) branch
186
+ target_branch.attributes[:head_commit_id] = merge_commit_id
187
+
188
+ update_checksum!
189
+ merge_commit
190
+ end
191
+
192
+ # Cherry-picks an existing commit onto the current branch.
193
+ # Creates a new commit on the current branch that mirrors the specified commit.
194
+ #
195
+ # @param commit_id [String] The ID of the commit to cherry-pick.
196
+ # @param parent_override_id [String, nil] Optional: If cherry-picking a merge commit, specifies which parent lineage to follow.
197
+ # (Note: Basic implementation might ignore this for simplicity initially).
198
+ # @raise [ArgumentError] if the commit_id does not exist, is already on the current branch,
199
+ # or if the current branch has no commits.
200
+ # @return [Elements::GitCommit] The created cherry-pick commit object.
201
+ # Basic implementation ignores parent_override_id for now
202
+ def cherry_pick(commit_id:, parent_override_id: nil)
203
+ unless @commits.key?(commit_id)
204
+ raise ArgumentError,
205
+ "Commit with ID '#{commit_id}' does not exist. Cannot cherry-pick."
206
+ end
207
+
208
+ source_commit = @commits[commit_id]
209
+ current_branch_head_id = current_head_commit_id
210
+
211
+ unless current_branch_head_id
212
+ raise ArgumentError,
213
+ "Current branch '#{@current_branch_name}' has no commits. Cannot cherry-pick onto it."
214
+ end
215
+ if source_commit.branch_name == @current_branch_name
216
+ raise ArgumentError,
217
+ "Commit '#{commit_id}' is already on the current branch '#{@current_branch_name}'. Cannot cherry-pick."
218
+ end
219
+
220
+ # More robust check: walk history? For now, simple branch name check.
221
+
222
+ # TODO: Handle cherry-picking merge commits and parent_override_id if needed later.
223
+ if source_commit.parent_ids.length > 1 && !parent_override_id
224
+ warn "Cherry-picking a merge commit (#{commit_id}) without specifying a parent override is ambiguous. Picking first parent lineage by default."
225
+ # Or raise ArgumentError: "Cherry-picking a merge commit requires specifying parent_override_id."
226
+ end
227
+
228
+ parent_ids = [current_branch_head_id] # Cherry-pick commit's parent is the current head
229
+ new_commit_id = generate_commit_id(parent_ids, "Cherry-pick: #{source_commit.message || source_commit.id}")
230
+ if @commits.key?(new_commit_id)
231
+ raise ArgumentError,
232
+ "Generated commit ID '#{new_commit_id}' conflicts with existing commit."
233
+ end
234
+
235
+ cherry_pick_commit = Elements::GitCommit.new(
236
+ id: new_commit_id,
237
+ parent_ids:,
238
+ branch_name: @current_branch_name,
239
+ message: source_commit.message || "Cherry-pick of #{source_commit.id}", # Copy message or use default
240
+ tag: nil, # Cherry-picks usually don't copy tags directly
241
+ type: :CHERRY_PICK,
242
+ cherry_pick_source_id: commit_id # Link back to the original commit
243
+ )
244
+
245
+ @commits[new_commit_id] = cherry_pick_commit
246
+ @commit_order << new_commit_id
247
+
248
+ # Update the head of the current branch
249
+ current_branch = @branches[@current_branch_name]
250
+ current_branch.attributes[:head_commit_id] = new_commit_id
251
+
252
+ update_checksum!
253
+ cherry_pick_commit
254
+ end
255
+
256
+ # --- Base Class Implementation ---
257
+
258
+ # Returns the specific content of the gitgraph diagram as a hash.
259
+ # @return [Hash]
260
+ def to_h_content
261
+ {
262
+ commits: @commits.values.map(&:to_h),
263
+ branches: @branches.values.map(&:to_h),
264
+ commit_order: @commit_order,
265
+ current_branch_name: @current_branch_name # Useful for resuming state? Maybe not needed in content hash.
266
+ # Consider if current_branch_name should be part of the checksummable content.
267
+ # For now, let's include it for potential deserialization needs.
268
+ }
269
+ end
270
+
271
+ # Returns a hash mapping element types to their collections for diffing.
272
+ # @return [Hash{Symbol => Array<Elements::GitCommit | Elements::GitBranch>}]
273
+ def identifiable_elements
274
+ {
275
+ commits: @commits.values,
276
+ branches: @branches.values
277
+ }
278
+ end
279
+
280
+ # Class method to create a GitgraphDiagram from a hash.
281
+ # @param data_hash [Hash] Hash containing diagram data.
282
+ # @param version [String, Integer, nil] Diagram version.
283
+ # @param checksum [String, nil] Expected checksum (optional).
284
+ # @return [GitgraphDiagram] The instantiated diagram.
285
+ def self.from_h(data_hash, version:, checksum:)
286
+ diagram = new(version:)
287
+
288
+ # Restore commits
289
+ commits_data = data_hash[:commits] || data_hash['commits'] || []
290
+ commits_data.each do |commit_h|
291
+ # Convert type back to symbol if it's a string
292
+ commit_data = commit_h.transform_keys(&:to_sym)
293
+ commit_data[:type] = commit_data[:type].to_sym if commit_data[:type].is_a?(String)
294
+ commit = Elements::GitCommit.new(commit_data)
295
+ diagram.commits[commit.id] = commit
296
+ end
297
+
298
+ # Restore branches
299
+ branches_data = data_hash[:branches] || data_hash['branches'] || []
300
+ branches_data.each do |branch_h|
301
+ branch = Elements::GitBranch.new(branch_h.transform_keys(&:to_sym))
302
+ diagram.branches[branch.name] = branch
303
+ end
304
+
305
+ # Restore commit order
306
+ diagram.instance_variable_set(:@commit_order, data_hash[:commit_order] || data_hash['commit_order'] || [])
307
+
308
+ # Restore current branch name
309
+ diagram.instance_variable_set(:@current_branch_name,
310
+ data_hash[:current_branch_name] || data_hash['current_branch_name'] || 'master')
311
+
312
+ # Recalculate checksum after loading all data
313
+ diagram.send(:update_checksum!) # Use send to call protected method from class scope
314
+
315
+ # Optional: Verify checksum if provided
316
+ if checksum && diagram.checksum != checksum
317
+ warn "Checksum mismatch for loaded GitgraphDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
318
+ end
319
+
320
+ diagram
321
+ end
322
+
323
+ private
324
+
325
+ # Generates a unique ID for a commit if one isn't provided.
326
+ # Placeholder - could use SHA1/SHA256 of content or simple counter.
327
+ # Using a simple counter based on commit order for now.
328
+ def generate_commit_id(parent_ids, message)
329
+ # Simple approach: use commit count + part of parent hash if available
330
+ base = @commit_order.size.to_s
331
+ parent_part = parent_ids.first ? parent_ids.first[0..5] : 'root'
332
+ # NOTE: This is NOT cryptographically secure or git-like. Just for basic uniqueness.
333
+ "commit-#{base}-#{parent_part}-#{Digest::SHA1.hexdigest(message || Time.now.to_s)[0..5]}"
334
+ end
335
+
336
+ # Helper to get the head commit ID of the current branch.
337
+ def current_head_commit_id
338
+ current_branch = @branches[@current_branch_name]
339
+ current_branch&.head_commit_id # Returns nil if branch doesn't exist yet
340
+ end
341
+
342
+ # Protected method access for from_h
343
+ protected :update_checksum!
344
+ end
345
+ end
@@ -1,8 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/slice'
5
-
6
3
  module Diagrams
7
4
  # Represents a Pie Chart diagram consisting of slices.
8
5
  class PieDiagram < Base
@@ -1,10 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/state'
5
- require_relative 'elements/transition'
6
- require_relative 'elements/event' # Assuming events are still desired
7
-
8
3
  module Diagrams
9
4
  # Represents a State Diagram consisting of states and transitions between them.
10
5
  class StateDiagram < Base