diagram 0.2.1 → 0.3.2
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/lib/diagram.rb +3 -0
- data/lib/diagrams/base.rb +244 -0
- data/lib/diagrams/class_diagram.rb +123 -2
- data/lib/diagrams/elements/class_entity.rb +24 -0
- data/lib/diagrams/elements/edge.rb +28 -0
- data/lib/diagrams/elements/event.rb +22 -0
- data/lib/diagrams/elements/git_branch.rb +27 -0
- data/lib/diagrams/elements/git_commit.rb +36 -0
- data/lib/diagrams/elements/node.rb +30 -0
- data/lib/diagrams/elements/relationship.rb +33 -0
- data/lib/diagrams/elements/slice.rb +23 -0
- data/lib/diagrams/elements/state.rb +24 -0
- data/lib/diagrams/elements/task.rb +23 -0
- data/lib/diagrams/elements/timeline_event.rb +21 -0
- data/lib/diagrams/elements/timeline_period.rb +23 -0
- data/lib/diagrams/elements/timeline_section.rb +23 -0
- data/lib/diagrams/elements/transition.rb +27 -0
- data/lib/diagrams/elements.rb +12 -0
- data/lib/diagrams/flowchart_diagram.rb +119 -4
- data/lib/diagrams/gantt_diagram.rb +94 -3
- data/lib/diagrams/gitgraph_diagram.rb +345 -0
- data/lib/diagrams/pie_diagram.rb +108 -29
- data/lib/diagrams/state_diagram.rb +157 -4
- data/lib/diagrams/timeline_diagram.rb +161 -0
- data/lib/diagrams/version.rb +1 -1
- data/lib/diagrams.rb +6 -1
- data/sig/diagrams/base.rbs +86 -0
- data/sig/diagrams/class_diagram.rbs +33 -0
- data/sig/diagrams/elements/class_entity.rbs +15 -0
- data/sig/diagrams/elements/edge.rbs +16 -0
- data/sig/diagrams/elements/event.rbs +14 -0
- data/sig/diagrams/elements/git_branch.rbs +19 -0
- data/sig/diagrams/elements/git_commit.rbs +23 -0
- data/sig/diagrams/elements/node.rbs +14 -0
- data/sig/diagrams/elements/relationship.rbs +16 -0
- data/sig/diagrams/elements/slice.rbs +14 -0
- data/sig/diagrams/elements/state.rbs +14 -0
- data/sig/diagrams/elements/task.rbs +16 -0
- data/sig/diagrams/elements/timeline_event.rbs +17 -0
- data/sig/diagrams/elements/timeline_period.rbs +18 -0
- data/sig/diagrams/elements/timeline_section.rbs +18 -0
- data/sig/diagrams/elements/transition.rbs +15 -0
- data/sig/diagrams/elements/types.rbs +18 -0
- data/sig/diagrams/flowchart_diagram.rbs +32 -0
- data/sig/diagrams/gantt_diagram.rbs +29 -0
- data/sig/diagrams/gitgraph_diagram.rbs +35 -0
- data/sig/diagrams/pie_diagram.rbs +36 -0
- data/sig/diagrams/state_diagram.rbs +40 -0
- data/sig/diagrams/timeline_diagram.rbs +33 -0
- metadata +228 -22
- data/lib/diagrams/abstract_diagram.rb +0 -26
- data/lib/diagrams/class_diagram/class/field.rb +0 -15
- data/lib/diagrams/class_diagram/class/function/argument.rb +0 -14
- data/lib/diagrams/class_diagram/class/function.rb +0 -16
- data/lib/diagrams/class_diagram/class.rb +0 -12
- data/lib/diagrams/comparable.rb +0 -20
- data/lib/diagrams/flowchart_diagram/link.rb +0 -11
- data/lib/diagrams/flowchart_diagram/node.rb +0 -10
- data/lib/diagrams/gantt_diagram/section/task.rb +0 -13
- data/lib/diagrams/gantt_diagram/section.rb +0 -10
- data/lib/diagrams/pie_diagram/section.rb +0 -10
- data/lib/diagrams/plot.rb +0 -23
- data/lib/diagrams/state_diagram/event.rb +0 -10
- data/lib/diagrams/state_diagram/state.rb +0 -12
- data/lib/diagrams/state_diagram/transition.rb +0 -11
@@ -0,0 +1,21 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a single event description within a timeline period.
|
6
|
+
class TimelineEvent < Dry::Struct
|
7
|
+
include Elements::Types
|
8
|
+
|
9
|
+
attribute :description, Types::Strict::String.constrained(min_size: 1)
|
10
|
+
|
11
|
+
# Returns a hash representation suitable for serialization.
|
12
|
+
#
|
13
|
+
# @return [Hash{Symbol => String}]
|
14
|
+
def to_h
|
15
|
+
{
|
16
|
+
description:
|
17
|
+
}
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a specific time period on the timeline, containing one or more events.
|
6
|
+
class TimelinePeriod < Dry::Struct
|
7
|
+
include Elements::Types
|
8
|
+
|
9
|
+
attribute :label, Types::Strict::String.constrained(min_size: 1)
|
10
|
+
attribute :events, Types::Strict::Array.of(TimelineEvent).constrained(min_size: 1)
|
11
|
+
|
12
|
+
# Returns a hash representation suitable for serialization.
|
13
|
+
#
|
14
|
+
# @return [Hash{Symbol => String | Array<Hash>}]
|
15
|
+
def to_h
|
16
|
+
{
|
17
|
+
label:,
|
18
|
+
events: events.map(&:to_h)
|
19
|
+
}
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a section or age within the timeline, grouping multiple time periods.
|
6
|
+
class TimelineSection < Dry::Struct
|
7
|
+
include Elements::Types
|
8
|
+
|
9
|
+
attribute :title, Types::Strict::String.constrained(min_size: 1)
|
10
|
+
attribute :periods, Types::Strict::Array.of(TimelinePeriod).default([].freeze)
|
11
|
+
|
12
|
+
# Returns a hash representation suitable for serialization.
|
13
|
+
#
|
14
|
+
# @return [Hash{Symbol => String | Array<Hash>}]
|
15
|
+
def to_h
|
16
|
+
{
|
17
|
+
title:,
|
18
|
+
periods: periods.map(&:to_h)
|
19
|
+
}
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a transition between two states in a State Diagram.
|
6
|
+
class Transition < Dry::Struct
|
7
|
+
# Use the shared Types module
|
8
|
+
include Elements::Types
|
9
|
+
|
10
|
+
# Consider if a transition needs its own ID.
|
11
|
+
# attribute :id, Types::Strict::String
|
12
|
+
|
13
|
+
attribute :source_state_id, Types::Strict::String.constrained(min_size: 1)
|
14
|
+
attribute :target_state_id, Types::Strict::String.constrained(min_size: 1)
|
15
|
+
# Label often represents the event or condition triggering the transition.
|
16
|
+
attribute :label, Types::Strict::String.optional.default(nil)
|
17
|
+
|
18
|
+
# Returns a hash representation suitable for serialization.
|
19
|
+
#
|
20
|
+
# @return [Hash{Symbol => String | nil}]
|
21
|
+
def to_h
|
22
|
+
# Rely on Dry::Struct's default to_h, filtering out nil label.
|
23
|
+
super.compact
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -1,9 +1,124 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Diagrams
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
4
|
+
# Represents a flowchart diagram consisting of nodes and edges connecting them.
|
5
|
+
class FlowchartDiagram < Base
|
6
|
+
attr_reader :nodes, :edges
|
7
|
+
|
8
|
+
# Initializes a new FlowchartDiagram.
|
9
|
+
#
|
10
|
+
# @param nodes [Array<Element::Node>] An array of node objects.
|
11
|
+
# @param edges [Array<Element::Edge>] An array of edge objects.
|
12
|
+
# @param version [String, Integer, nil] User-defined version identifier.
|
13
|
+
def initialize(nodes: [], edges: [], version: 1)
|
14
|
+
super(version:)
|
15
|
+
@nodes = nodes.is_a?(Array) ? nodes : []
|
16
|
+
@edges = edges.is_a?(Array) ? edges : []
|
17
|
+
validate_elements!
|
18
|
+
update_checksum!
|
19
|
+
end
|
20
|
+
|
21
|
+
# Adds a node to the diagram.
|
22
|
+
#
|
23
|
+
# @param node [Element::Node] The node object to add.
|
24
|
+
# @raise [ArgumentError] if a node with the same ID already exists.
|
25
|
+
# @return [Element::Node] The added node.
|
26
|
+
def add_node(node)
|
27
|
+
raise ArgumentError, 'Node must be a Diagrams::Element::Node' unless node.is_a?(Diagrams::Elements::Node)
|
28
|
+
raise ArgumentError, "Node with ID '#{node.id}' already exists" if find_node(node.id)
|
29
|
+
|
30
|
+
@nodes << node
|
31
|
+
update_checksum!
|
32
|
+
node
|
33
|
+
end
|
34
|
+
|
35
|
+
# Adds an edge to the diagram.
|
36
|
+
#
|
37
|
+
# @param edge [Element::Edge] The edge object to add.
|
38
|
+
# @raise [ArgumentError] if the edge refers to non-existent node IDs.
|
39
|
+
# @return [Element::Edge] The added edge.
|
40
|
+
def add_edge(edge)
|
41
|
+
raise ArgumentError, 'Edge must be a Diagrams::Element::Edge' unless edge.is_a?(Diagrams::Elements::Edge)
|
42
|
+
unless find_node(edge.source_id) && find_node(edge.target_id)
|
43
|
+
raise ArgumentError, "Edge refers to non-existent node IDs ('#{edge.source_id}' or '#{edge.target_id}')"
|
44
|
+
end
|
45
|
+
|
46
|
+
@edges << edge
|
47
|
+
update_checksum!
|
48
|
+
edge
|
49
|
+
end
|
50
|
+
|
51
|
+
# Finds a node by its ID.
|
52
|
+
#
|
53
|
+
# @param node_id [String] The ID of the node to find.
|
54
|
+
# @return [Element::Node, nil] The found node or nil.
|
55
|
+
def find_node(node_id)
|
56
|
+
@nodes.find { |n| n.id == node_id }
|
57
|
+
end
|
58
|
+
|
59
|
+
# Returns the specific content of the flowchart diagram as a hash.
|
60
|
+
# Called by `Diagrams::Base#to_h`.
|
61
|
+
#
|
62
|
+
# @return [Hash{Symbol => Array<Hash>}]
|
63
|
+
def to_h_content
|
64
|
+
{
|
65
|
+
nodes: @nodes.map(&:to_h),
|
66
|
+
edges: @edges.map(&:to_h)
|
67
|
+
}
|
68
|
+
end
|
69
|
+
|
70
|
+
# Returns a hash mapping element types to their collections for diffing.
|
71
|
+
# @see Diagrams::Base#identifiable_elements
|
72
|
+
# @return [Hash{Symbol => Array<Diagrams::Elements::Node | Diagrams::Elements::Edge>}]
|
73
|
+
def identifiable_elements
|
74
|
+
{
|
75
|
+
nodes: @nodes,
|
76
|
+
edges: @edges
|
77
|
+
}
|
78
|
+
end
|
79
|
+
|
80
|
+
# Class method to create a FlowchartDiagram from a hash.
|
81
|
+
# Used by the deserialization factory in `Diagrams::Base`.
|
82
|
+
#
|
83
|
+
# @param data_hash [Hash] Hash containing `:nodes` and `:edges` arrays.
|
84
|
+
# @param version [String, Integer, nil] Diagram version.
|
85
|
+
# @param checksum [String, nil] Expected checksum (optional, for verification).
|
86
|
+
# @return [FlowchartDiagram] The instantiated diagram.
|
87
|
+
def self.from_h(data_hash, version:, checksum:)
|
88
|
+
nodes_data = data_hash[:nodes] || data_hash['nodes'] || []
|
89
|
+
edges_data = data_hash[:edges] || data_hash['edges'] || []
|
90
|
+
|
91
|
+
nodes =
|
92
|
+
nodes_data.map do |node_h|
|
93
|
+
Diagrams::Elements::Node.new(node_h.transform_keys(&:to_sym))
|
94
|
+
end
|
95
|
+
edges =
|
96
|
+
edges_data.map do |edge_h|
|
97
|
+
Diagrams::Elements::Edge.new(edge_h.transform_keys(&:to_sym))
|
98
|
+
end
|
99
|
+
diagram = new(nodes:, edges:, version:)
|
100
|
+
|
101
|
+
# Optional: Verify checksum if provided
|
102
|
+
if checksum && diagram.checksum != checksum
|
103
|
+
warn "Checksum mismatch for loaded FlowchartDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
|
104
|
+
# Or raise an error: raise "Checksum mismatch..."
|
105
|
+
end
|
106
|
+
|
107
|
+
diagram
|
108
|
+
end
|
109
|
+
|
110
|
+
private
|
111
|
+
|
112
|
+
# Validates the consistency of nodes and edges during initialization.
|
113
|
+
def validate_elements!
|
114
|
+
node_ids = @nodes.map(&:id)
|
115
|
+
raise ArgumentError, 'Duplicate node IDs found' unless node_ids.uniq.size == @nodes.size
|
116
|
+
|
117
|
+
@edges.each do |edge|
|
118
|
+
unless node_ids.include?(edge.source_id) && node_ids.include?(edge.target_id)
|
119
|
+
raise ArgumentError, "Edge refers to non-existent node IDs ('#{edge.source_id}' or '#{edge.target_id}')"
|
120
|
+
end
|
121
|
+
end
|
122
|
+
end
|
8
123
|
end
|
9
124
|
end
|
@@ -1,8 +1,99 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Diagrams
|
4
|
-
|
5
|
-
|
6
|
-
|
4
|
+
# Represents a Gantt Chart diagram consisting of tasks over time.
|
5
|
+
class GanttDiagram < Base
|
6
|
+
attr_reader :title, :tasks
|
7
|
+
|
8
|
+
# Initializes a new GanttDiagram.
|
9
|
+
#
|
10
|
+
# @param title [String] The title of the Gantt chart.
|
11
|
+
# @param tasks [Array<Element::Task>] An array of task objects.
|
12
|
+
# @param version [String, Integer, nil] User-defined version identifier.
|
13
|
+
def initialize(title: '', tasks: [], version: 1)
|
14
|
+
super(version:)
|
15
|
+
@title = title.is_a?(String) ? title : ''
|
16
|
+
@tasks = tasks.is_a?(Array) ? tasks : []
|
17
|
+
validate_elements!
|
18
|
+
update_checksum!
|
19
|
+
end
|
20
|
+
|
21
|
+
# Adds a task to the diagram.
|
22
|
+
#
|
23
|
+
# @param task [Element::Task] The task object to add.
|
24
|
+
# @raise [ArgumentError] if a task with the same ID already exists.
|
25
|
+
# @return [Element::Task] The added task.
|
26
|
+
def add_task(task)
|
27
|
+
raise ArgumentError, 'Task must be a Diagrams::Elements::Task' unless task.is_a?(Diagrams::Elements::Task)
|
28
|
+
raise ArgumentError, "Task with ID '#{task.id}' already exists" if find_task(task.id)
|
29
|
+
|
30
|
+
@tasks << task
|
31
|
+
update_checksum!
|
32
|
+
task
|
33
|
+
end
|
34
|
+
|
35
|
+
# Finds a task by its ID.
|
36
|
+
#
|
37
|
+
# @param task_id [String] The ID of the task to find.
|
38
|
+
# @return [Element::Task, nil] The found task or nil.
|
39
|
+
def find_task(task_id)
|
40
|
+
@tasks.find { |t| t.id == task_id }
|
41
|
+
end
|
42
|
+
|
43
|
+
# Returns the specific content of the Gantt diagram as a hash.
|
44
|
+
# Called by `Diagrams::Base#to_h`.
|
45
|
+
#
|
46
|
+
# @return [Hash{Symbol => String | Array<Hash>}]
|
47
|
+
def to_h_content
|
48
|
+
{
|
49
|
+
title: @title,
|
50
|
+
tasks: @tasks.map(&:to_h)
|
51
|
+
}
|
52
|
+
end
|
53
|
+
|
54
|
+
# Returns a hash mapping element types to their collections for diffing.
|
55
|
+
# @see Diagrams::Base#identifiable_elements
|
56
|
+
# @return [Hash{Symbol => Array<Diagrams::Elements::Task>}]
|
57
|
+
def identifiable_elements
|
58
|
+
{
|
59
|
+
tasks: @tasks
|
60
|
+
}
|
61
|
+
end
|
62
|
+
|
63
|
+
# Class method to create a GanttDiagram from a hash.
|
64
|
+
# Used by the deserialization factory in `Diagrams::Base`.
|
65
|
+
#
|
66
|
+
# @param data_hash [Hash] Hash containing `:title` and `:tasks` array.
|
67
|
+
# @param version [String, Integer, nil] Diagram version.
|
68
|
+
# @param checksum [String, nil] Expected checksum (optional, for verification).
|
69
|
+
# @return [GanttDiagram] The instantiated diagram.
|
70
|
+
def self.from_h(data_hash, version:, checksum:)
|
71
|
+
title = data_hash[:title] || data_hash['title'] || ''
|
72
|
+
tasks_data = data_hash[:tasks] || data_hash['tasks'] || []
|
73
|
+
|
74
|
+
tasks = tasks_data.map { |task_h| Diagrams::Elements::Task.new(task_h.transform_keys(&:to_sym)) }
|
75
|
+
|
76
|
+
diagram = new(title:, tasks:, version:)
|
77
|
+
|
78
|
+
# Optional: Verify checksum if provided
|
79
|
+
if checksum && diagram.checksum != checksum
|
80
|
+
warn "Checksum mismatch for loaded GanttDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
|
81
|
+
# Or raise an error: raise "Checksum mismatch..."
|
82
|
+
end
|
83
|
+
|
84
|
+
diagram
|
85
|
+
end
|
86
|
+
|
87
|
+
private
|
88
|
+
|
89
|
+
# Validates the consistency of tasks during initialization.
|
90
|
+
def validate_elements!
|
91
|
+
task_ids = @tasks.map(&:id)
|
92
|
+
return if task_ids.uniq.size == @tasks.size
|
93
|
+
|
94
|
+
raise ArgumentError, 'Duplicate task IDs found'
|
95
|
+
|
96
|
+
# Add more validation if needed (e.g., date formats, dependencies)
|
97
|
+
end
|
7
98
|
end
|
8
99
|
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
|