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.
Files changed (66) hide show
  1. checksums.yaml +4 -4
  2. data/lib/diagram.rb +3 -0
  3. data/lib/diagrams/base.rb +244 -0
  4. data/lib/diagrams/class_diagram.rb +123 -2
  5. data/lib/diagrams/elements/class_entity.rb +24 -0
  6. data/lib/diagrams/elements/edge.rb +28 -0
  7. data/lib/diagrams/elements/event.rb +22 -0
  8. data/lib/diagrams/elements/git_branch.rb +27 -0
  9. data/lib/diagrams/elements/git_commit.rb +36 -0
  10. data/lib/diagrams/elements/node.rb +30 -0
  11. data/lib/diagrams/elements/relationship.rb +33 -0
  12. data/lib/diagrams/elements/slice.rb +23 -0
  13. data/lib/diagrams/elements/state.rb +24 -0
  14. data/lib/diagrams/elements/task.rb +23 -0
  15. data/lib/diagrams/elements/timeline_event.rb +21 -0
  16. data/lib/diagrams/elements/timeline_period.rb +23 -0
  17. data/lib/diagrams/elements/timeline_section.rb +23 -0
  18. data/lib/diagrams/elements/transition.rb +27 -0
  19. data/lib/diagrams/elements.rb +12 -0
  20. data/lib/diagrams/flowchart_diagram.rb +119 -4
  21. data/lib/diagrams/gantt_diagram.rb +94 -3
  22. data/lib/diagrams/gitgraph_diagram.rb +345 -0
  23. data/lib/diagrams/pie_diagram.rb +108 -29
  24. data/lib/diagrams/state_diagram.rb +157 -4
  25. data/lib/diagrams/timeline_diagram.rb +161 -0
  26. data/lib/diagrams/version.rb +1 -1
  27. data/lib/diagrams.rb +6 -1
  28. data/sig/diagrams/base.rbs +86 -0
  29. data/sig/diagrams/class_diagram.rbs +33 -0
  30. data/sig/diagrams/elements/class_entity.rbs +15 -0
  31. data/sig/diagrams/elements/edge.rbs +16 -0
  32. data/sig/diagrams/elements/event.rbs +14 -0
  33. data/sig/diagrams/elements/git_branch.rbs +19 -0
  34. data/sig/diagrams/elements/git_commit.rbs +23 -0
  35. data/sig/diagrams/elements/node.rbs +14 -0
  36. data/sig/diagrams/elements/relationship.rbs +16 -0
  37. data/sig/diagrams/elements/slice.rbs +14 -0
  38. data/sig/diagrams/elements/state.rbs +14 -0
  39. data/sig/diagrams/elements/task.rbs +16 -0
  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/elements/transition.rbs +15 -0
  44. data/sig/diagrams/elements/types.rbs +18 -0
  45. data/sig/diagrams/flowchart_diagram.rbs +32 -0
  46. data/sig/diagrams/gantt_diagram.rbs +29 -0
  47. data/sig/diagrams/gitgraph_diagram.rbs +35 -0
  48. data/sig/diagrams/pie_diagram.rbs +36 -0
  49. data/sig/diagrams/state_diagram.rbs +40 -0
  50. data/sig/diagrams/timeline_diagram.rbs +33 -0
  51. metadata +228 -22
  52. data/lib/diagrams/abstract_diagram.rb +0 -26
  53. data/lib/diagrams/class_diagram/class/field.rb +0 -15
  54. data/lib/diagrams/class_diagram/class/function/argument.rb +0 -14
  55. data/lib/diagrams/class_diagram/class/function.rb +0 -16
  56. data/lib/diagrams/class_diagram/class.rb +0 -12
  57. data/lib/diagrams/comparable.rb +0 -20
  58. data/lib/diagrams/flowchart_diagram/link.rb +0 -11
  59. data/lib/diagrams/flowchart_diagram/node.rb +0 -10
  60. data/lib/diagrams/gantt_diagram/section/task.rb +0 -13
  61. data/lib/diagrams/gantt_diagram/section.rb +0 -10
  62. data/lib/diagrams/pie_diagram/section.rb +0 -10
  63. data/lib/diagrams/plot.rb +0 -23
  64. data/lib/diagrams/state_diagram/event.rb +0 -10
  65. data/lib/diagrams/state_diagram/state.rb +0 -12
  66. data/lib/diagrams/state_diagram/transition.rb +0 -11
@@ -1,48 +1,127 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Diagrams
4
- class PieDiagram < AbstractDiagram
5
- attribute :id, Types::String.optional.default(nil)
6
- attribute :title, Types::String.optional.default('')
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
- def type
10
- :pie
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
- def validate!
14
- raise EmptyDiagramError, 'Pie diagram must have at least one section' if sections.empty?
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
- return true if sections.map(&:label).uniq.size == sections.size
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
- raise DuplicateLabelError,
19
- 'Pie diagram sections must have unique labels'
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
- def plot
23
- circle_char = '*'
24
- pie_diameter = 10
25
- pie_radius = pie_diameter / 2.0
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
- (-pie_radius.to_i..pie_radius.to_i).each do |i|
28
- (-pie_radius.to_i..pie_radius.to_i).each do |j|
29
- distance_to_center = Math.sqrt((i**2) + (j**2))
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
- sections.each do |section|
40
- puts "#{section.label}: #{section.value}%"
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
- def valid?
45
- validate!
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
- class StateDiagram < AbstractDiagram
5
- attribute :id, Types::String
6
- attribute :states, Types::Array.of(State)
7
- attribute :transitions, Types::Array.of(Transition)
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
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Diagrams
4
- VERSION = '0.2.1'
4
+ VERSION = '0.3.2'
5
5
  end
data/lib/diagrams.rb CHANGED
@@ -1,9 +1,14 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'diagrams/version'
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