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
data/lib/diagrams/pie_diagram.rb
CHANGED
@@ -1,48 +1,127 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Diagrams
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
attribute :sections, Types::Array.of(Section).optional.default([].freeze)
|
4
|
+
# Represents a Pie Chart diagram consisting of slices.
|
5
|
+
class PieDiagram < Base
|
6
|
+
attr_reader :title, :slices
|
8
7
|
|
9
|
-
|
10
|
-
|
8
|
+
# Initializes a new PieDiagram.
|
9
|
+
#
|
10
|
+
# @param title [String] The title of the pie chart.
|
11
|
+
# @param slices [Array<Element::Slice>] An array of slice objects.
|
12
|
+
# @param version [String, Integer, nil] User-defined version identifier.
|
13
|
+
def initialize(title: '', slices: [], version: 1)
|
14
|
+
super(version: version) # Corrected super call
|
15
|
+
@title = title.is_a?(String) ? title : ''
|
16
|
+
@slices = [] # Initialize empty
|
17
|
+
# Add initial slices using the corrected add_slice method
|
18
|
+
(slices.is_a?(Array) ? slices : []).each { |s| add_slice(s, update_checksum: false, initial_load: true) }
|
19
|
+
recalculate_percentages! # Calculate initial percentages
|
20
|
+
update_checksum! # Calculate final checksum after initial load
|
11
21
|
end
|
12
22
|
|
13
|
-
|
14
|
-
|
23
|
+
# Adds a slice to the diagram.
|
24
|
+
#
|
25
|
+
# @param slice [Element::Slice] The slice object to add.
|
26
|
+
# @raise [ArgumentError] if a slice with the same label already exists.
|
27
|
+
# @return [Element::Slice] The added slice.
|
28
|
+
# Added initial_load flag to skip checksum update during initialize loop
|
29
|
+
def add_slice(slice, update_checksum: true, initial_load: false)
|
30
|
+
raise ArgumentError, 'Slice must be a Diagrams::Elements::Slice' unless slice.is_a?(Diagrams::Elements::Slice)
|
31
|
+
raise ArgumentError, "Slice with label '#{slice.label}' already exists" if find_slice(slice.label)
|
15
32
|
|
16
|
-
|
33
|
+
# Store a new instance to hold the calculated percentage later
|
34
|
+
# Ensure percentage is nil initially
|
35
|
+
new_slice_instance = slice.class.new(slice.attributes.except(:percentage))
|
36
|
+
@slices << new_slice_instance
|
37
|
+
recalculate_percentages! # Update percentages for all slices
|
38
|
+
update_checksum! if update_checksum && !initial_load # Avoid multiple checksums during init
|
39
|
+
new_slice_instance # Return the instance added to the array
|
40
|
+
end
|
41
|
+
|
42
|
+
# Finds a slice by its label.
|
43
|
+
#
|
44
|
+
# @param label [String] The label of the slice to find.
|
45
|
+
# @return [Element::Slice, nil] The found slice or nil.
|
46
|
+
def find_slice(label)
|
47
|
+
@slices.find { |s| s.label == label }
|
48
|
+
end
|
49
|
+
|
50
|
+
# Calculates the total raw value of all slices.
|
51
|
+
# @return [Float]
|
52
|
+
def total_value
|
53
|
+
@slices.sum(&:value)
|
54
|
+
end
|
55
|
+
|
56
|
+
# Returns the specific content of the pie diagram as a hash.
|
57
|
+
# Called by `Diagrams::Base#to_h`.
|
58
|
+
#
|
59
|
+
# @return [Hash{Symbol => String | Array<Hash>}]
|
60
|
+
def to_h_content
|
61
|
+
{
|
62
|
+
title: @title,
|
63
|
+
# Ensure slices include calculated percentage in their hash
|
64
|
+
slices: @slices.map(&:to_h)
|
65
|
+
}
|
66
|
+
end
|
17
67
|
|
18
|
-
|
19
|
-
|
68
|
+
# Returns a hash mapping element types to their collections for diffing.
|
69
|
+
# @see Diagrams::Base#identifiable_elements
|
70
|
+
# @return [Hash{Symbol => Array<Diagrams::Elements::Slice>}]
|
71
|
+
def identifiable_elements
|
72
|
+
{
|
73
|
+
slices: @slices
|
74
|
+
}
|
20
75
|
end
|
21
76
|
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
77
|
+
# Class method to create a PieDiagram from a hash.
|
78
|
+
# Used by the deserialization factory in `Diagrams::Base`.
|
79
|
+
#
|
80
|
+
# @param data_hash [Hash] Hash containing `:title` and `:slices` array.
|
81
|
+
# @param version [String, Integer, nil] Diagram version.
|
82
|
+
# @param checksum [String, nil] Expected checksum (optional, for verification).
|
83
|
+
# @return [PieDiagram] The instantiated diagram.
|
84
|
+
def self.from_h(data_hash, version:, checksum:)
|
85
|
+
title = data_hash[:title] || data_hash['title'] || ''
|
86
|
+
slices_data = data_hash[:slices] || data_hash['slices'] || []
|
26
87
|
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
if distance_to_center > pie_radius - 0.5 && distance_to_center < pie_radius + 0.5
|
31
|
-
print circle_char
|
32
|
-
else
|
33
|
-
print ' '
|
34
|
-
end
|
35
|
-
end
|
36
|
-
print "\n"
|
88
|
+
# Initialize with raw values, percentage will be recalculated by `new` -> `add_slice` -> `recalculate_percentages!`
|
89
|
+
slices = slices_data.map do |slice_h|
|
90
|
+
Diagrams::Elements::Slice.new(slice_h.transform_keys(&:to_sym).except(:percentage))
|
37
91
|
end
|
38
92
|
|
39
|
-
|
40
|
-
|
93
|
+
diagram = new(title: title, slices: slices, version: version)
|
94
|
+
|
95
|
+
# Optional: Verify checksum if provided AFTER initialization is complete
|
96
|
+
if checksum && diagram.checksum != checksum
|
97
|
+
warn "Checksum mismatch for loaded PieDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
|
98
|
+
# Or raise an error: raise "Checksum mismatch..."
|
41
99
|
end
|
100
|
+
|
101
|
+
diagram
|
42
102
|
end
|
43
103
|
|
44
|
-
|
45
|
-
|
104
|
+
private
|
105
|
+
|
106
|
+
# Recalculates the percentage for each slice based on the total value.
|
107
|
+
# This method modifies the @slices array in place by replacing Slice instances.
|
108
|
+
def recalculate_percentages!
|
109
|
+
total = total_value
|
110
|
+
new_slices = @slices.map do |slice|
|
111
|
+
percentage = total.zero? ? 0.0 : (slice.value / total * 100.0).round(2)
|
112
|
+
# Create a new instance with the calculated percentage
|
113
|
+
slice.class.new(slice.attributes.merge(percentage: percentage))
|
114
|
+
end
|
115
|
+
# Replace the entire array to ensure changes are reflected
|
116
|
+
@slices = new_slices
|
117
|
+
end
|
118
|
+
|
119
|
+
# Validates the consistency of slices during initialization.
|
120
|
+
def validate_elements!
|
121
|
+
labels = @slices.map(&:label)
|
122
|
+
return if labels.uniq.size == @slices.size
|
123
|
+
|
124
|
+
raise ArgumentError, 'Duplicate slice labels found'
|
46
125
|
end
|
47
126
|
end
|
48
127
|
end
|
@@ -1,9 +1,162 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Diagrams
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
4
|
+
# Represents a State Diagram consisting of states and transitions between them.
|
5
|
+
class StateDiagram < Base
|
6
|
+
attr_reader :title, :states, :transitions, :events
|
7
|
+
|
8
|
+
# Initializes a new StateDiagram.
|
9
|
+
#
|
10
|
+
# @param title [String] Optional title for the diagram.
|
11
|
+
# @param states [Array<Element::State>] An array of state objects.
|
12
|
+
# @param transitions [Array<Element::Transition>] An array of transition objects.
|
13
|
+
# @param events [Array<Element::Event>] An array of event objects (optional).
|
14
|
+
# @param version [String, Integer, nil] User-defined version identifier.
|
15
|
+
def initialize(title: '', states: [], transitions: [], events: [], version: 1)
|
16
|
+
super(version:)
|
17
|
+
@title = title.is_a?(String) ? title : ''
|
18
|
+
@states = states.is_a?(Array) ? states : []
|
19
|
+
@transitions = transitions.is_a?(Array) ? transitions : []
|
20
|
+
@events = events.is_a?(Array) ? events : [] # Keep events for now
|
21
|
+
validate_elements!
|
22
|
+
update_checksum!
|
23
|
+
end
|
24
|
+
|
25
|
+
# Adds a state to the diagram.
|
26
|
+
#
|
27
|
+
# @param state [Element::State] The state object to add.
|
28
|
+
# @raise [ArgumentError] if a state with the same ID already exists.
|
29
|
+
# @return [Element::State] The added state.
|
30
|
+
def add_state(state)
|
31
|
+
raise ArgumentError, 'State must be a Diagrams::Elements::State' unless state.is_a?(Diagrams::Elements::State)
|
32
|
+
raise ArgumentError, "State with ID '#{state.id}' already exists" if find_state(state.id)
|
33
|
+
|
34
|
+
@states << state
|
35
|
+
update_checksum!
|
36
|
+
state
|
37
|
+
end
|
38
|
+
|
39
|
+
# Adds a transition to the diagram.
|
40
|
+
#
|
41
|
+
# @param transition [Element::Transition] The transition object to add.
|
42
|
+
# @raise [ArgumentError] if the transition refers to non-existent state IDs.
|
43
|
+
# @return [Element::Transition] The added transition.
|
44
|
+
def add_transition(transition)
|
45
|
+
unless transition.is_a?(Diagrams::Elements::Transition)
|
46
|
+
raise ArgumentError,
|
47
|
+
'Transition must be a Diagrams::Elements::Transition'
|
48
|
+
end
|
49
|
+
unless find_state(transition.source_state_id) && find_state(transition.target_state_id)
|
50
|
+
raise ArgumentError,
|
51
|
+
"Transition refers to non-existent state IDs ('#{transition.source_state_id}' or '#{transition.target_state_id}')"
|
52
|
+
end
|
53
|
+
|
54
|
+
@transitions << transition
|
55
|
+
update_checksum!
|
56
|
+
transition
|
57
|
+
end
|
58
|
+
|
59
|
+
# Adds an event to the diagram.
|
60
|
+
#
|
61
|
+
# @param event [Element::Event] The event object to add.
|
62
|
+
# @raise [ArgumentError] if an event with the same ID already exists.
|
63
|
+
# @return [Element::Event] The added event.
|
64
|
+
def add_event(event)
|
65
|
+
raise ArgumentError, 'Event must be a Diagrams::Elements::Event' unless event.is_a?(Diagrams::Elements::Event)
|
66
|
+
raise ArgumentError, "Event with ID '#{event.id}' already exists" if find_event(event.id)
|
67
|
+
|
68
|
+
@events << event
|
69
|
+
update_checksum!
|
70
|
+
event
|
71
|
+
end
|
72
|
+
|
73
|
+
# Finds a state by its ID.
|
74
|
+
#
|
75
|
+
# @param state_id [String] The ID of the state to find.
|
76
|
+
# @return [Element::State, nil] The found state or nil.
|
77
|
+
def find_state(state_id)
|
78
|
+
@states.find { |s| s.id == state_id }
|
79
|
+
end
|
80
|
+
|
81
|
+
# Finds an event by its ID.
|
82
|
+
#
|
83
|
+
# @param event_id [String] The ID of the event to find.
|
84
|
+
# @return [Element::Event, nil] The found event or nil.
|
85
|
+
def find_event(event_id)
|
86
|
+
@events.find { |e| e.id == event_id }
|
87
|
+
end
|
88
|
+
|
89
|
+
# Returns the specific content of the state diagram as a hash.
|
90
|
+
# Called by `Diagrams::Base#to_h`.
|
91
|
+
#
|
92
|
+
# @return [Hash{Symbol => String | Array<Hash>}]
|
93
|
+
def to_h_content
|
94
|
+
{
|
95
|
+
title: @title,
|
96
|
+
states: @states.map(&:to_h),
|
97
|
+
transitions: @transitions.map(&:to_h),
|
98
|
+
events: @events.map(&:to_h)
|
99
|
+
}
|
100
|
+
end
|
101
|
+
|
102
|
+
# Returns a hash mapping element types to their collections for diffing.
|
103
|
+
# @see Diagrams::Base#identifiable_elements
|
104
|
+
# @return [Hash{Symbol => Array<Diagrams::Elements::State | Diagrams::Elements::Transition | Diagrams::Elements::Event>}]
|
105
|
+
def identifiable_elements
|
106
|
+
{
|
107
|
+
states: @states,
|
108
|
+
transitions: @transitions,
|
109
|
+
events: @events
|
110
|
+
}
|
111
|
+
end
|
112
|
+
|
113
|
+
# Class method to create a StateDiagram from a hash.
|
114
|
+
# Used by the deserialization factory in `Diagrams::Base`.
|
115
|
+
#
|
116
|
+
# @param data_hash [Hash] Hash containing `:title`, `:states`, `:transitions`, `:events`.
|
117
|
+
# @param version [String, Integer, nil] Diagram version.
|
118
|
+
# @param checksum [String, nil] Expected checksum (optional, for verification).
|
119
|
+
# @return [StateDiagram] The instantiated diagram.
|
120
|
+
def self.from_h(data_hash, version:, checksum:)
|
121
|
+
title = data_hash[:title] || data_hash['title'] || ''
|
122
|
+
states_data = data_hash[:states] || data_hash['states'] || []
|
123
|
+
transitions_data = data_hash[:transitions] || data_hash['transitions'] || []
|
124
|
+
events_data = data_hash[:events] || data_hash['events'] || []
|
125
|
+
|
126
|
+
states = states_data.map { |state_h| Diagrams::Elements::State.new(state_h.transform_keys(&:to_sym)) }
|
127
|
+
transitions = transitions_data.map do |trans_h|
|
128
|
+
Diagrams::Elements::Transition.new(trans_h.transform_keys(&:to_sym))
|
129
|
+
end
|
130
|
+
events = events_data.map { |event_h| Diagrams::Elements::Event.new(event_h.transform_keys(&:to_sym)) }
|
131
|
+
|
132
|
+
diagram = new(title:, states:, transitions:, events:, version:)
|
133
|
+
|
134
|
+
# Optional: Verify checksum if provided
|
135
|
+
if checksum && diagram.checksum != checksum
|
136
|
+
warn "Checksum mismatch for loaded StateDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
|
137
|
+
# Or raise an error: raise "Checksum mismatch..."
|
138
|
+
end
|
139
|
+
|
140
|
+
diagram
|
141
|
+
end
|
142
|
+
|
143
|
+
private
|
144
|
+
|
145
|
+
# Validates the consistency of elements during initialization.
|
146
|
+
def validate_elements!
|
147
|
+
state_ids = @states.map(&:id)
|
148
|
+
raise ArgumentError, 'Duplicate state IDs found' unless state_ids.uniq.size == @states.size
|
149
|
+
|
150
|
+
event_ids = @events.map(&:id)
|
151
|
+
raise ArgumentError, 'Duplicate event IDs found' unless event_ids.uniq.size == @events.size
|
152
|
+
|
153
|
+
@transitions.each do |t|
|
154
|
+
unless state_ids.include?(t.source_state_id) && state_ids.include?(t.target_state_id)
|
155
|
+
raise ArgumentError,
|
156
|
+
"Transition refers to non-existent state IDs ('#{t.source_state_id}' or '#{t.target_state_id}')"
|
157
|
+
end
|
158
|
+
end
|
159
|
+
# Add more validation if needed (e.g., transition labels match event IDs?)
|
160
|
+
end
|
8
161
|
end
|
9
162
|
end
|
@@ -0,0 +1,161 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
# Represents a timeline diagram illustrating a chronology of events.
|
5
|
+
class TimelineDiagram < Base
|
6
|
+
DEFAULT_SECTION_TITLE = 'Default Section'
|
7
|
+
|
8
|
+
attr_reader :title, :sections
|
9
|
+
|
10
|
+
# Initializes a new TimelineDiagram.
|
11
|
+
#
|
12
|
+
# @param title [String, nil] Optional title for the timeline.
|
13
|
+
# @param sections [Array<Element::TimelineSection>] Initial sections (optional).
|
14
|
+
# @param version [String, Integer, nil] User-defined version identifier.
|
15
|
+
def initialize(title: nil, sections: [], version: 1)
|
16
|
+
super(version:)
|
17
|
+
@title = title&.strip
|
18
|
+
@sections = sections.is_a?(Array) ? sections : []
|
19
|
+
# Ensure there's always at least a default section if none provided initially
|
20
|
+
ensure_default_section if @sections.empty?
|
21
|
+
update_checksum!
|
22
|
+
end
|
23
|
+
|
24
|
+
# Sets the title of the timeline.
|
25
|
+
#
|
26
|
+
# @param new_title [String] The title text.
|
27
|
+
# @return [String] The new title.
|
28
|
+
def set_title(new_title)
|
29
|
+
@title = new_title.strip
|
30
|
+
update_checksum!
|
31
|
+
@title
|
32
|
+
end
|
33
|
+
|
34
|
+
# Adds a new section to the timeline.
|
35
|
+
# Subsequent periods/events will be added to this section.
|
36
|
+
#
|
37
|
+
# @param section_title [String] The title of the section.
|
38
|
+
# @raise [ArgumentError] if a section with the same title already exists.
|
39
|
+
# @return [Elements::TimelineSection] The newly added section.
|
40
|
+
def add_section(section_title)
|
41
|
+
clean_title = section_title.strip
|
42
|
+
raise ArgumentError, "Section title '#{clean_title}' cannot be empty" if clean_title.empty?
|
43
|
+
if find_section(clean_title)
|
44
|
+
raise ArgumentError, "Section with title '#{clean_title}' already exists"
|
45
|
+
end
|
46
|
+
|
47
|
+
# Remove default section if it's empty and we're adding a real one
|
48
|
+
if @sections.size == 1 && @sections.first.title == DEFAULT_SECTION_TITLE && @sections.first.periods.empty?
|
49
|
+
@sections.clear
|
50
|
+
end
|
51
|
+
|
52
|
+
new_section = Elements::TimelineSection.new(title: clean_title)
|
53
|
+
@sections << new_section
|
54
|
+
update_checksum!
|
55
|
+
new_section
|
56
|
+
end
|
57
|
+
|
58
|
+
# Adds a time period with one or more events to the current (last) section.
|
59
|
+
#
|
60
|
+
# @param period_label [String] The label for the time period (e.g., "2004", "Bronze Age").
|
61
|
+
# @param events [Array<String> | String] A single event description or an array of event descriptions.
|
62
|
+
# @raise [ArgumentError] if period_label or any event description is empty.
|
63
|
+
# @raise [StandardError] if no sections exist (shouldn't happen due to default section).
|
64
|
+
# @return [Elements::TimelinePeriod] The newly added period.
|
65
|
+
def add_period(period_label:, events:)
|
66
|
+
clean_label = period_label.strip
|
67
|
+
raise ArgumentError, "Period label cannot be empty" if clean_label.empty?
|
68
|
+
|
69
|
+
event_list = Array(events).map(&:strip).reject(&:empty?)
|
70
|
+
raise ArgumentError, "Events cannot be empty" if event_list.empty?
|
71
|
+
|
72
|
+
timeline_events = event_list.map { |desc| Elements::TimelineEvent.new(description: desc) }
|
73
|
+
new_period = Elements::TimelinePeriod.new(label: clean_label, events: timeline_events)
|
74
|
+
|
75
|
+
current_section = @sections.last
|
76
|
+
raise StandardError, "Cannot add period: No section available." unless current_section
|
77
|
+
|
78
|
+
# Add period to the current section's periods array
|
79
|
+
# Dry::Struct arrays are immutable, so we need to create a new section object
|
80
|
+
updated_periods = current_section.periods + [new_period]
|
81
|
+
# Create a completely new section instance with the updated periods array
|
82
|
+
updated_section = Elements::TimelineSection.new(title: current_section.title, periods: updated_periods)
|
83
|
+
|
84
|
+
# Find the index of the current section and update it in place
|
85
|
+
# Rebuild the sections array, replacing the modified section
|
86
|
+
current_section_title = current_section.title
|
87
|
+
# Rebuild the sections array by mapping, replacing the target section
|
88
|
+
@sections = @sections.map do |section|
|
89
|
+
section.title == current_section_title ? updated_section : section
|
90
|
+
end
|
91
|
+
|
92
|
+
update_checksum!
|
93
|
+
new_period
|
94
|
+
end
|
95
|
+
|
96
|
+
# --- Base Class Implementation ---
|
97
|
+
|
98
|
+
def to_h_content
|
99
|
+
content = {
|
100
|
+
sections: @sections.map(&:to_h)
|
101
|
+
}
|
102
|
+
content[:title] = @title if @title
|
103
|
+
content
|
104
|
+
end
|
105
|
+
|
106
|
+
def identifiable_elements
|
107
|
+
# Sections and Periods are the main identifiable structures. Events are nested.
|
108
|
+
# Use section title and period label as identifiers.
|
109
|
+
{
|
110
|
+
sections: @sections,
|
111
|
+
periods: @sections.flat_map(&:periods) # Flatten periods from all sections
|
112
|
+
}
|
113
|
+
end
|
114
|
+
|
115
|
+
def self.from_h(data_hash, version:, checksum:)
|
116
|
+
title = data_hash[:title] || data_hash['title']
|
117
|
+
sections_data = data_hash[:sections] || data_hash['sections'] || []
|
118
|
+
|
119
|
+
sections = sections_data.map do |section_h|
|
120
|
+
section_data = section_h.transform_keys(&:to_sym)
|
121
|
+
periods_data = section_data[:periods] || []
|
122
|
+
periods = periods_data.map do |period_h|
|
123
|
+
period_data = period_h.transform_keys(&:to_sym)
|
124
|
+
events_data = period_data[:events] || []
|
125
|
+
events = events_data.map do |event_h|
|
126
|
+
event_data = event_h.transform_keys(&:to_sym)
|
127
|
+
Elements::TimelineEvent.new(event_data)
|
128
|
+
end
|
129
|
+
Elements::TimelinePeriod.new(period_data.merge(events:))
|
130
|
+
end
|
131
|
+
Elements::TimelineSection.new(section_data.merge(periods:))
|
132
|
+
end
|
133
|
+
|
134
|
+
diagram = new(title:, sections:, version:)
|
135
|
+
|
136
|
+
# Optional: Verify checksum
|
137
|
+
if checksum && diagram.checksum != checksum
|
138
|
+
warn "Checksum mismatch for loaded TimelineDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
|
139
|
+
end
|
140
|
+
|
141
|
+
diagram
|
142
|
+
end
|
143
|
+
|
144
|
+
private
|
145
|
+
|
146
|
+
# Ensures a default section exists if the sections array is empty.
|
147
|
+
def ensure_default_section
|
148
|
+
unless @sections.any? { |s| s.title == DEFAULT_SECTION_TITLE }
|
149
|
+
@sections << Elements::TimelineSection.new(title: DEFAULT_SECTION_TITLE)
|
150
|
+
end
|
151
|
+
end
|
152
|
+
|
153
|
+
# Finds a section by its title.
|
154
|
+
def find_section(section_title)
|
155
|
+
@sections.find { |s| s.title == section_title }
|
156
|
+
end
|
157
|
+
|
158
|
+
# Protected method access for from_h
|
159
|
+
protected :update_checksum!
|
160
|
+
end
|
161
|
+
end
|
data/lib/diagrams/version.rb
CHANGED
data/lib/diagrams.rb
CHANGED
@@ -1,9 +1,14 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
3
|
+
require 'zeitwerk'
|
4
|
+
require 'digest'
|
5
|
+
require 'json'
|
6
|
+
require 'dry-equalizer'
|
4
7
|
require 'dry-struct'
|
8
|
+
require_relative 'diagrams/version'
|
5
9
|
|
6
10
|
loader = Zeitwerk::Loader.for_gem
|
11
|
+
loader.ignore("#{__dir__}/diagram.rb")
|
7
12
|
loader.setup
|
8
13
|
|
9
14
|
# This module handles diagrams creation and manipulation.
|
@@ -0,0 +1,86 @@
|
|
1
|
+
module Diagrams
|
2
|
+
# Abstract base class for all diagram types.
|
3
|
+
# Provides common functionality like versioning, checksum calculation,
|
4
|
+
# serialization, and equality comparison.
|
5
|
+
class Base
|
6
|
+
# Provides `==`, `eql?`, and `hash` methods based on specified attributes.
|
7
|
+
include Dry::Equalizer
|
8
|
+
|
9
|
+
# User-defined version identifier (e.g., Integer or String).
|
10
|
+
attr_reader version: Integer | String
|
11
|
+
|
12
|
+
# SHA256 checksum of the diagram's content (hex string). Nil until calculated.
|
13
|
+
attr_reader checksum: String?
|
14
|
+
|
15
|
+
# Initializes the base diagram attributes.
|
16
|
+
# Subclasses should call super.
|
17
|
+
# Cannot be called directly on Diagrams::Base.
|
18
|
+
def initialize: (?version: Integer | String?) -> void
|
19
|
+
|
20
|
+
# Abstract method: Subclasses must implement this to return a hash
|
21
|
+
# representing their specific content, suitable for serialization.
|
22
|
+
# @return [Hash[Symbol, untyped]]
|
23
|
+
def to_h_content: () -> Hash[Symbol, untyped]
|
24
|
+
|
25
|
+
# Abstract method: Subclasses must implement this to return a hash
|
26
|
+
# mapping element type symbols (e.g., :nodes, :edges) to arrays
|
27
|
+
# of the corresponding element objects within the diagram.
|
28
|
+
# Used for comparison and diffing.
|
29
|
+
# @return [Hash[Symbol, Array[untyped]]] # More specific types in subclasses
|
30
|
+
def identifiable_elements: () -> Hash[Symbol, Array[untyped]]
|
31
|
+
|
32
|
+
# Returns a hash representation of the diagram, suitable for serialization.
|
33
|
+
# Includes common metadata and calls `#to_h_content` for specific data.
|
34
|
+
# @return [Hash[Symbol, untyped]]
|
35
|
+
def to_h: () -> Hash[Symbol, untyped]
|
36
|
+
|
37
|
+
# Returns a JSON string representation of the diagram.
|
38
|
+
# Delegates to `#to_h` and uses `JSON.generate`.
|
39
|
+
# @param args Any arguments accepted by `JSON.generate`.
|
40
|
+
# @return [String]
|
41
|
+
def to_json: (*untyped args) -> String
|
42
|
+
|
43
|
+
# --- Class methods for Deserialization ---
|
44
|
+
|
45
|
+
# Deserializes a diagram from a hash representation.
|
46
|
+
# Acts as a factory, dispatching to the appropriate subclass based on the 'type' field.
|
47
|
+
# @param hash [Hash[Symbol | String, untyped]] The hash representation.
|
48
|
+
# @return [Diagrams::Base] An instance of the specific diagram subclass.
|
49
|
+
def self.from_hash: (Hash[Symbol | String, untyped] hash) -> Diagrams::Base
|
50
|
+
|
51
|
+
# Deserializes a diagram from a JSON string.
|
52
|
+
# Parses the JSON and delegates to `.from_hash`.
|
53
|
+
# @param json_string [String] The JSON representation of the diagram.
|
54
|
+
# @return [Diagrams::Base] An instance of the specific diagram subclass.
|
55
|
+
def self.from_json: (String json_string) -> Diagrams::Base
|
56
|
+
|
57
|
+
# Performs a basic diff against another diagram object.
|
58
|
+
# @param other [Diagrams::Base] The diagram to compare against.
|
59
|
+
# @return [Hash[Symbol, Hash[Symbol, Array[untyped]]]] A hash describing differences.
|
60
|
+
def diff: (Diagrams::Base other) -> Hash[Symbol, Hash[Symbol, Array[untyped]]]
|
61
|
+
|
62
|
+
# --- End Deserialization Methods ---
|
63
|
+
|
64
|
+
# Recalculates the diagram's checksum based on its current content
|
65
|
+
# and updates the @checksum instance variable.
|
66
|
+
# Subclasses should call this after initialization and any mutation.
|
67
|
+
# @return [String?] The new checksum value.
|
68
|
+
def update_checksum!: () -> String?
|
69
|
+
|
70
|
+
private
|
71
|
+
|
72
|
+
# Computes the SHA256 checksum of the diagram's content.
|
73
|
+
# @return [String] The hex digest of the checksum.
|
74
|
+
def compute_checksum: () -> String
|
75
|
+
end
|
76
|
+
|
77
|
+
# --- Errors ---
|
78
|
+
class ValidationError < StandardError
|
79
|
+
end
|
80
|
+
|
81
|
+
class EmptyDiagramError < ValidationError
|
82
|
+
end
|
83
|
+
|
84
|
+
class DuplicateLabelError < ValidationError
|
85
|
+
end
|
86
|
+
end
|
@@ -0,0 +1,33 @@
|
|
1
|
+
|
2
|
+
module Diagrams
|
3
|
+
class ClassDiagram < Base
|
4
|
+
attr_reader classes: ::Array[Elements::ClassEntity]
|
5
|
+
attr_reader relationships: ::Array[Elements::Relationship]
|
6
|
+
|
7
|
+
# Initializes a new ClassDiagram.
|
8
|
+
def initialize: (?classes: ::Array[Elements::ClassEntity]?, ?relationships: ::Array[Elements::Relationship]?, ?version: Integer | String?) -> void
|
9
|
+
|
10
|
+
# Adds a class entity to the diagram.
|
11
|
+
def add_class: (Elements::ClassEntity class_entity) -> Elements::ClassEntity
|
12
|
+
|
13
|
+
# Adds a relationship to the diagram.
|
14
|
+
def add_relationship: (Elements::Relationship relationship) -> Elements::Relationship
|
15
|
+
|
16
|
+
# Finds a class entity by its name.
|
17
|
+
def find_class: (::String class_name) -> Elements::ClassEntity?
|
18
|
+
|
19
|
+
# Returns the specific content of the class diagram as a hash.
|
20
|
+
def to_h_content: () -> { classes: ::Array[Hash[Symbol, untyped]], relationships: ::Array[Hash[Symbol, untyped]] }
|
21
|
+
|
22
|
+
# Returns a hash mapping element types to their collections for diffing.
|
23
|
+
def identifiable_elements: () -> { classes: ::Array[Elements::ClassEntity], relationships: ::Array[Elements::Relationship] }
|
24
|
+
|
25
|
+
# Class method to create a ClassDiagram from a hash.
|
26
|
+
def self.from_h: (Hash[Symbol | String, untyped] data_hash, version: Integer | String?, checksum: String?) -> ClassDiagram
|
27
|
+
|
28
|
+
private
|
29
|
+
|
30
|
+
# Validates the consistency of classes and relationships during initialization.
|
31
|
+
def validate_elements!: () -> void
|
32
|
+
end
|
33
|
+
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
module Diagrams
|
2
|
+
module Elements
|
3
|
+
class ClassEntity < ::Dry::Struct
|
4
|
+
include Diagrams::Elements::Types
|
5
|
+
|
6
|
+
# Attributes
|
7
|
+
def name: () -> ::String
|
8
|
+
def attributes: () -> ::Array[::String]
|
9
|
+
def methods: () -> ::Array[::String]
|
10
|
+
|
11
|
+
# Methods
|
12
|
+
def to_h: () -> { name: ::String, attributes: ::Array[::String], methods: ::Array[::String] }
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module Diagrams
|
2
|
+
module Elements
|
3
|
+
class Edge < ::Dry::Struct
|
4
|
+
include Diagrams::Elements::Types
|
5
|
+
|
6
|
+
# Attributes
|
7
|
+
def source_id: () -> ::String
|
8
|
+
def target_id: () -> ::String
|
9
|
+
def label: () -> ::String?
|
10
|
+
|
11
|
+
# Methods
|
12
|
+
# Dry::Struct provides to_h, signature reflects potential nil label removal
|
13
|
+
def to_h: () -> { source_id: ::String, target_id: ::String, ?label: ::String }
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
@@ -0,0 +1,14 @@
|
|
1
|
+
module Diagrams
|
2
|
+
module Elements
|
3
|
+
class Event < ::Dry::Struct
|
4
|
+
include Diagrams::Elements::Types
|
5
|
+
|
6
|
+
# Attributes
|
7
|
+
def id: () -> ::String
|
8
|
+
def label: () -> ::String?
|
9
|
+
|
10
|
+
# Methods
|
11
|
+
def to_h: () -> { id: ::String, ?label: ::String }
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|