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
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: b6a7329e68d37cab7fe076fb763b56fddff42b23e81619bd70f3a89ef82ebff8
|
4
|
+
data.tar.gz: 3eac3513c7ebb76307967f12f3c9d189e35022203ce817e3fbc90be310ac7b10
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 0b59a02a496318c080f8ff72bbfb6bb224ae4dfeb3657eda3e262ac401cca6871f8a18a487fbe012699c7016ae8ef07cfc565f224e7697cc3a6aba9abb60907b
|
7
|
+
data.tar.gz: 6cee2aa87b44ae4aab58a99fcf9cfadfbde2f9496523d7e02e5dcb00f6c4ec14abce8b36ff79c1b276e44eec407fe09db6dcaabd9a14c83a050b7684df854f41
|
data/lib/diagram.rb
ADDED
@@ -0,0 +1,244 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
# Abstract base class for all diagram types.
|
5
|
+
# Provides common functionality like versioning, checksum calculation,
|
6
|
+
# serialization, and equality comparison.
|
7
|
+
class Base
|
8
|
+
# Provides `==`, `eql?`, and `hash` methods based on specified attributes.
|
9
|
+
# Diagrams are equal if they are of the same class and have the same content (checksum).
|
10
|
+
include Dry::Equalizer(:class, :checksum)
|
11
|
+
|
12
|
+
attr_reader :version, :checksum
|
13
|
+
|
14
|
+
# Initializes the base diagram attributes.
|
15
|
+
# Subclasses should call super.
|
16
|
+
#
|
17
|
+
# @param version [String, Integer, nil] User-defined version identifier. Defaults to 1.
|
18
|
+
def initialize(version: 1)
|
19
|
+
# Prevent direct instantiation of the base class
|
20
|
+
raise NotImplementedError, 'Cannot instantiate abstract class Diagrams::Base' if instance_of?(Diagrams::Base)
|
21
|
+
|
22
|
+
@version = version
|
23
|
+
@checksum = nil # Will be calculated by subclasses via #update_checksum! after content is set
|
24
|
+
end
|
25
|
+
|
26
|
+
# Abstract method: Subclasses must implement this to return a hash
|
27
|
+
# representing their specific content, suitable for serialization.
|
28
|
+
#
|
29
|
+
# @return [Hash]
|
30
|
+
def to_h_content
|
31
|
+
raise NotImplementedError, "#{self.class.name} must implement #to_h_content"
|
32
|
+
end
|
33
|
+
|
34
|
+
# Abstract method: Subclasses must implement this to return a hash
|
35
|
+
# mapping element type symbols (e.g., :nodes, :edges) to arrays
|
36
|
+
# of the corresponding element objects within the diagram.
|
37
|
+
# Used for comparison and diffing.
|
38
|
+
#
|
39
|
+
# @return [Hash{Symbol => Array<Diagrams::Elements::*>}]
|
40
|
+
def identifiable_elements
|
41
|
+
raise NotImplementedError, "#{self.class.name} must implement #identifiable_elements"
|
42
|
+
end
|
43
|
+
|
44
|
+
# Performs a basic diff against another diagram object.
|
45
|
+
# Only compares diagrams of the same type.
|
46
|
+
# Identifies added and removed elements based on common identifiers (id/name) or object equality.
|
47
|
+
# Does NOT currently detect modified elements.
|
48
|
+
#
|
49
|
+
# @param other [Diagrams::Base] The diagram to compare against.
|
50
|
+
# @return [Hash{Symbol => Hash{Symbol => Array<Diagrams::Elements::*>}}] A hash describing differences,
|
51
|
+
# e.g., { nodes: { added: [...], removed: [...] }, edges: { added: [...], removed: [...] } }
|
52
|
+
# Returns an empty hash if diagrams are identical or of different types.
|
53
|
+
def diff(other)
|
54
|
+
diff_result = {}
|
55
|
+
return diff_result unless other.is_a?(self.class) # Only compare same types
|
56
|
+
return diff_result if self == other # Use existing equality check for quick exit
|
57
|
+
|
58
|
+
self_elements = identifiable_elements
|
59
|
+
other_elements = other.identifiable_elements
|
60
|
+
|
61
|
+
# Ensure both diagrams define the same element types for comparison
|
62
|
+
element_types = self_elements.keys & other_elements.keys
|
63
|
+
|
64
|
+
element_types.each do |type|
|
65
|
+
self_collection = self_elements[type] || []
|
66
|
+
other_collection = other_elements[type] || []
|
67
|
+
|
68
|
+
# Determine identifier method (prefer id, then name, then title, then label, fallback to object itself)
|
69
|
+
identifier_method = if self_collection.first&.respond_to?(:id)
|
70
|
+
:id
|
71
|
+
elsif self_collection.first&.respond_to?(:name)
|
72
|
+
:name
|
73
|
+
elsif self_collection.first&.respond_to?(:title) # For TimelineSection
|
74
|
+
:title
|
75
|
+
elsif self_collection.first&.respond_to?(:label) # For Slice, TimelinePeriod
|
76
|
+
:label
|
77
|
+
else
|
78
|
+
:itself # Fallback to object identity/equality
|
79
|
+
end
|
80
|
+
|
81
|
+
self_ids = self_collection.map(&identifier_method)
|
82
|
+
other_ids = other_collection.map(&identifier_method)
|
83
|
+
|
84
|
+
added_ids = other_ids - self_ids
|
85
|
+
removed_ids = self_ids - other_ids
|
86
|
+
|
87
|
+
added_elements = other_collection.select { |el| added_ids.include?(el.send(identifier_method)) }
|
88
|
+
removed_elements = self_collection.select { |el| removed_ids.include?(el.send(identifier_method)) }
|
89
|
+
|
90
|
+
# Basic check for modified elements (same ID, different content via checksum/hash if available, or simple !=)
|
91
|
+
# This is a very basic modification check
|
92
|
+
potential_modified_ids = self_ids & other_ids
|
93
|
+
modified_elements = []
|
94
|
+
potential_modified_ids.each do |id|
|
95
|
+
self_el = self_collection.find { |el| el.send(identifier_method) == id }
|
96
|
+
other_el = other_collection.find { |el| el.send(identifier_method) == id }
|
97
|
+
# Use Dry::Struct equality if available, otherwise basic !=
|
98
|
+
next unless self_el != other_el
|
99
|
+
|
100
|
+
modified_elements << { old: self_el, new: other_el }
|
101
|
+
# Remove from added/removed if detected as modified
|
102
|
+
added_elements.delete(other_el)
|
103
|
+
removed_elements.delete(self_el)
|
104
|
+
end
|
105
|
+
|
106
|
+
type_diff = {}
|
107
|
+
type_diff[:added] = added_elements if added_elements.any?
|
108
|
+
type_diff[:removed] = removed_elements if removed_elements.any?
|
109
|
+
type_diff[:modified] = modified_elements if modified_elements.any? # Add modified info
|
110
|
+
|
111
|
+
diff_result[type] = type_diff if type_diff.any?
|
112
|
+
end
|
113
|
+
|
114
|
+
diff_result
|
115
|
+
end
|
116
|
+
|
117
|
+
# Returns a hash representation of the diagram, suitable for serialization.
|
118
|
+
# Includes common metadata and calls `#to_h_content` for specific data.
|
119
|
+
#
|
120
|
+
# @return [Hash]
|
121
|
+
def to_h
|
122
|
+
{
|
123
|
+
# Extract class name without module prefix (e.g., "FlowchartDiagram")
|
124
|
+
# Convert class name to snake_case (e.g., FlowchartDiagram -> flowchart_diagram)
|
125
|
+
type: camel_to_snake_case(self.class.name.split('::').last),
|
126
|
+
version: @version,
|
127
|
+
checksum: @checksum, # Ensure checksum is up-to-date before calling
|
128
|
+
data: to_h_content
|
129
|
+
}
|
130
|
+
end
|
131
|
+
|
132
|
+
# Returns a JSON string representation of the diagram.
|
133
|
+
# Delegates to `#to_h` and uses `JSON.generate`.
|
134
|
+
# Accepts any arguments valid for `JSON.generate`.
|
135
|
+
#
|
136
|
+
# @param _args Any arguments accepted by `JSON.generate` (ignored by method signature but passed along).
|
137
|
+
# @return [String]
|
138
|
+
def to_json(*)
|
139
|
+
JSON.generate(to_h, *)
|
140
|
+
end
|
141
|
+
|
142
|
+
# --- Class methods for Deserialization ---
|
143
|
+
|
144
|
+
class << self
|
145
|
+
# Deserializes a diagram from a hash representation.
|
146
|
+
# Acts as a factory, dispatching to the appropriate subclass based on the 'type' field.
|
147
|
+
#
|
148
|
+
# @param hash [Hash] The hash representation (typically from parsed JSON).
|
149
|
+
# @return [Diagrams::Base] An instance of the specific diagram subclass.
|
150
|
+
# @raise [ArgumentError] if the hash is missing the 'type' key.
|
151
|
+
# @raise [NameError] if the type string doesn't correspond to a known Diagram class.
|
152
|
+
# @raise [TypeError] if the resolved class is not a subclass of Diagrams::Base.
|
153
|
+
def from_hash(hash)
|
154
|
+
# Ensure keys are symbols for consistent access
|
155
|
+
symbolized_hash = hash.transform_keys(&:to_sym)
|
156
|
+
|
157
|
+
type_string = symbolized_hash[:type]
|
158
|
+
raise ArgumentError, "Input hash must include a 'type' key." unless type_string
|
159
|
+
|
160
|
+
data_hash = symbolized_hash[:data] || {}
|
161
|
+
version = symbolized_hash[:version]
|
162
|
+
checksum = symbolized_hash[:checksum] # Pass checksum for potential verification
|
163
|
+
|
164
|
+
begin
|
165
|
+
# Convert snake_case type string back to CamelCase class name part
|
166
|
+
camel_case_type = snake_to_camel_case(type_string)
|
167
|
+
# Construct full class name (e.g., "Diagrams::FlowchartDiagram")
|
168
|
+
klass_name = "Diagrams::#{camel_case_type}"
|
169
|
+
klass = Object.const_get(klass_name)
|
170
|
+
rescue NameError
|
171
|
+
raise NameError, "Unknown diagram type '#{type_string}' corresponding to class '#{klass_name}'"
|
172
|
+
end
|
173
|
+
|
174
|
+
# Ensure the resolved class is actually a diagram type
|
175
|
+
raise TypeError, "'#{klass_name}' is not a valid subclass of Diagrams::Base" unless klass < Diagrams::Base
|
176
|
+
|
177
|
+
# Delegate to the specific subclass's from_h method
|
178
|
+
# Each subclass must implement `from_h(data_hash, version:, checksum:)`
|
179
|
+
klass.from_h(data_hash, version:, checksum:)
|
180
|
+
end
|
181
|
+
|
182
|
+
# Deserializes a diagram from a JSON string.
|
183
|
+
# Parses the JSON and delegates to `.from_hash`.
|
184
|
+
#
|
185
|
+
# @param json_string [String] The JSON representation of the diagram.
|
186
|
+
# @return [Diagrams::Base] An instance of the specific diagram subclass.
|
187
|
+
def from_json(json_string)
|
188
|
+
hash = JSON.parse(json_string)
|
189
|
+
from_hash(hash)
|
190
|
+
rescue JSON::ParserError => e
|
191
|
+
raise JSON::ParserError, "Failed to parse JSON: #{e.message}"
|
192
|
+
end
|
193
|
+
|
194
|
+
private # Make helper private to the class methods
|
195
|
+
|
196
|
+
# Simple helper to convert snake_case to CamelCase
|
197
|
+
# (Avoids ActiveSupport dependency)
|
198
|
+
def snake_to_camel_case(string)
|
199
|
+
string.split('_').collect(&:capitalize).join
|
200
|
+
end
|
201
|
+
end
|
202
|
+
|
203
|
+
# --- End Deserialization Methods ---
|
204
|
+
|
205
|
+
protected
|
206
|
+
|
207
|
+
# Recalculates the diagram's checksum based on its current content
|
208
|
+
# and updates the @checksum instance variable.
|
209
|
+
# Subclasses should call this after initialization and any mutation.
|
210
|
+
def update_checksum!
|
211
|
+
@checksum = compute_checksum
|
212
|
+
end
|
213
|
+
|
214
|
+
# Simple helper to convert CamelCase to snake_case
|
215
|
+
# (Avoids ActiveSupport dependency)
|
216
|
+
def camel_to_snake_case(string)
|
217
|
+
string.gsub('::', '/')
|
218
|
+
.gsub(/([A-Z]+)([A-Z][a-z])/, '\1_\2')
|
219
|
+
.gsub(/([a-z\d])([A-Z])/, '\1_\2')
|
220
|
+
.tr('-', '_')
|
221
|
+
.downcase
|
222
|
+
end
|
223
|
+
|
224
|
+
# Computes the SHA256 checksum of the diagram's content.
|
225
|
+
# The content is obtained from `#to_h_content` and serialized to JSON
|
226
|
+
# to ensure a consistent representation for hashing.
|
227
|
+
#
|
228
|
+
# @return [String] The hex digest of the checksum.
|
229
|
+
def compute_checksum
|
230
|
+
# Ensure content is available before computing checksum
|
231
|
+
content_hash = respond_to?(:to_h_content, true) ? to_h_content : {}
|
232
|
+
# Generate JSON. Sorting keys isn't strictly necessary for SHA256
|
233
|
+
# but can help if comparing JSON strings directly elsewhere.
|
234
|
+
# For checksum purposes, consistency is key, which JSON.generate provides.
|
235
|
+
content_json = JSON.generate(content_hash || {}) # Handle potential nil from to_h_content
|
236
|
+
Digest::SHA256.hexdigest(content_json)
|
237
|
+
end
|
238
|
+
end
|
239
|
+
|
240
|
+
## Errors (Kept from original file)
|
241
|
+
class ValidationError < StandardError; end
|
242
|
+
class EmptyDiagramError < ValidationError; end
|
243
|
+
class DuplicateLabelError < ValidationError; end
|
244
|
+
end
|
@@ -1,7 +1,128 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Diagrams
|
4
|
-
|
5
|
-
|
4
|
+
# Represents a UML Class Diagram consisting of classes and relationships between them.
|
5
|
+
class ClassDiagram < Base
|
6
|
+
attr_reader :classes, :relationships
|
7
|
+
|
8
|
+
# Initializes a new ClassDiagram.
|
9
|
+
#
|
10
|
+
# @param classes [Array<Element::ClassEntity>] An array of class entity objects.
|
11
|
+
# @param relationships [Array<Element::Relationship>] An array of relationship objects.
|
12
|
+
# @param version [String, Integer, nil] User-defined version identifier.
|
13
|
+
def initialize(classes: [], relationships: [], version: 1)
|
14
|
+
super(version:)
|
15
|
+
@classes = classes.is_a?(Array) ? classes : []
|
16
|
+
@relationships = relationships.is_a?(Array) ? relationships : []
|
17
|
+
validate_elements!
|
18
|
+
update_checksum!
|
19
|
+
end
|
20
|
+
|
21
|
+
# Adds a class entity to the diagram.
|
22
|
+
#
|
23
|
+
# @param class_entity [Element::ClassEntity] The class entity object to add.
|
24
|
+
# @raise [ArgumentError] if a class with the same name already exists.
|
25
|
+
# @return [Element::ClassEntity] The added class entity.
|
26
|
+
def add_class(class_entity)
|
27
|
+
unless class_entity.is_a?(Diagrams::Elements::ClassEntity)
|
28
|
+
raise ArgumentError,
|
29
|
+
'Class entity must be a Diagrams::Elements::ClassEntity'
|
30
|
+
end
|
31
|
+
raise ArgumentError, "Class with name '#{class_entity.name}' already exists" if find_class(class_entity.name)
|
32
|
+
|
33
|
+
@classes << class_entity
|
34
|
+
update_checksum!
|
35
|
+
class_entity
|
36
|
+
end
|
37
|
+
|
38
|
+
# Adds a relationship to the diagram.
|
39
|
+
#
|
40
|
+
# @param relationship [Element::Relationship] The relationship object to add.
|
41
|
+
# @raise [ArgumentError] if the relationship refers to non-existent class names.
|
42
|
+
# @return [Element::Relationship] The added relationship.
|
43
|
+
def add_relationship(relationship)
|
44
|
+
unless relationship.is_a?(Diagrams::Elements::Relationship)
|
45
|
+
raise ArgumentError,
|
46
|
+
'Relationship must be a Diagrams::Elements::Relationship'
|
47
|
+
end
|
48
|
+
unless find_class(relationship.source_class_name) && find_class(relationship.target_class_name)
|
49
|
+
raise ArgumentError,
|
50
|
+
"Relationship refers to non-existent class names ('#{relationship.source_class_name}' or '#{relationship.target_class_name}')"
|
51
|
+
end
|
52
|
+
|
53
|
+
@relationships << relationship
|
54
|
+
update_checksum!
|
55
|
+
relationship
|
56
|
+
end
|
57
|
+
|
58
|
+
# Finds a class entity by its name.
|
59
|
+
#
|
60
|
+
# @param class_name [String] The name of the class to find.
|
61
|
+
# @return [Element::ClassEntity, nil] The found class entity or nil.
|
62
|
+
def find_class(class_name)
|
63
|
+
@classes.find { |c| c.name == class_name }
|
64
|
+
end
|
65
|
+
|
66
|
+
# Returns the specific content of the class diagram as a hash.
|
67
|
+
# Called by `Diagrams::Base#to_h`.
|
68
|
+
#
|
69
|
+
# @return [Hash{Symbol => Array<Hash>}]
|
70
|
+
def to_h_content
|
71
|
+
{
|
72
|
+
classes: @classes.map(&:to_h),
|
73
|
+
relationships: @relationships.map(&:to_h)
|
74
|
+
}
|
75
|
+
end
|
76
|
+
|
77
|
+
# Returns a hash mapping element types to their collections for diffing.
|
78
|
+
# @see Diagrams::Base#identifiable_elements
|
79
|
+
# @return [Hash{Symbol => Array<Diagrams::Elements::ClassEntity | Diagrams::Elements::Relationship>}]
|
80
|
+
def identifiable_elements
|
81
|
+
{
|
82
|
+
classes: @classes,
|
83
|
+
relationships: @relationships
|
84
|
+
}
|
85
|
+
end
|
86
|
+
|
87
|
+
# Class method to create a ClassDiagram from a hash.
|
88
|
+
# Used by the deserialization factory in `Diagrams::Base`.
|
89
|
+
#
|
90
|
+
# @param data_hash [Hash] Hash containing `:classes` and `:relationships` arrays.
|
91
|
+
# @param version [String, Integer, nil] Diagram version.
|
92
|
+
# @param checksum [String, nil] Expected checksum (optional, for verification).
|
93
|
+
# @return [ClassDiagram] The instantiated diagram.
|
94
|
+
def self.from_h(data_hash, version:, checksum:)
|
95
|
+
classes_data = data_hash[:classes] || data_hash['classes'] || []
|
96
|
+
relationships_data = data_hash[:relationships] || data_hash['relationships'] || []
|
97
|
+
|
98
|
+
classes = classes_data.map { |class_h| Diagrams::Elements::ClassEntity.new(class_h.transform_keys(&:to_sym)) }
|
99
|
+
relationships = relationships_data.map do |rel_h|
|
100
|
+
Diagrams::Elements::Relationship.new(rel_h.transform_keys(&:to_sym))
|
101
|
+
end
|
102
|
+
|
103
|
+
diagram = new(classes:, relationships:, version:)
|
104
|
+
|
105
|
+
if checksum && diagram.checksum != checksum
|
106
|
+
warn "Checksum mismatch for loaded ClassDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
|
107
|
+
# Or raise an error: raise "Checksum mismatch..."
|
108
|
+
end
|
109
|
+
|
110
|
+
diagram
|
111
|
+
end
|
112
|
+
|
113
|
+
private
|
114
|
+
|
115
|
+
# Validates the consistency of classes and relationships during initialization.
|
116
|
+
def validate_elements!
|
117
|
+
class_names = @classes.map(&:name)
|
118
|
+
raise ArgumentError, 'Duplicate class names found' unless class_names.uniq.size == @classes.size
|
119
|
+
|
120
|
+
@relationships.each do |rel|
|
121
|
+
unless class_names.include?(rel.source_class_name) && class_names.include?(rel.target_class_name)
|
122
|
+
raise ArgumentError,
|
123
|
+
"Relationship refers to non-existent class names ('#{rel.source_class_name}' or '#{rel.target_class_name}')"
|
124
|
+
end
|
125
|
+
end
|
126
|
+
end
|
6
127
|
end
|
7
128
|
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a class entity in a UML Class Diagram.
|
6
|
+
class ClassEntity < Dry::Struct
|
7
|
+
# Use the shared Types module
|
8
|
+
include Elements::Types
|
9
|
+
|
10
|
+
# Name of the class
|
11
|
+
attribute :name, Types::Strict::String.constrained(min_size: 1)
|
12
|
+
|
13
|
+
# List of attributes (e.g., "id: Integer", "name: String")
|
14
|
+
attribute :attributes, Types::Strict::Array.of(Types::Strict::String).default([].freeze)
|
15
|
+
|
16
|
+
# List of methods (e.g., "save()", "find(id: Integer)")
|
17
|
+
attribute :methods, Types::Strict::Array.of(Types::Strict::String).default([].freeze)
|
18
|
+
|
19
|
+
# Returns a hash representation suitable for serialization.
|
20
|
+
#
|
21
|
+
# @return [Hash{Symbol => String | Array<String>}]
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents an edge or link between two nodes in a diagram.
|
6
|
+
# Typically connects nodes via their IDs and can have an optional label.
|
7
|
+
class Edge < Dry::Struct
|
8
|
+
# Use the shared Types module defined in node.rb (or a dedicated types file)
|
9
|
+
include Elements::Types
|
10
|
+
|
11
|
+
# Consider if an edge needs its own ID, or if source/target/label is sufficient identity.
|
12
|
+
# attribute :id, Types::Strict::String
|
13
|
+
|
14
|
+
attribute :source_id, Types::Strict::String.constrained(min_size: 1)
|
15
|
+
attribute :target_id, Types::Strict::String.constrained(min_size: 1)
|
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, which includes all attributes.
|
23
|
+
# Filter out nil label if desired.
|
24
|
+
super.compact
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents an event, potentially used in State Diagrams or others.
|
6
|
+
class Event < Dry::Struct
|
7
|
+
# Use the shared Types module
|
8
|
+
include Elements::Types
|
9
|
+
|
10
|
+
attribute :id, Types::Strict::String.constrained(min_size: 1)
|
11
|
+
attribute :label, Types::Strict::String.optional.default(nil)
|
12
|
+
|
13
|
+
# Returns a hash representation suitable for serialization.
|
14
|
+
#
|
15
|
+
# @return [Hash{Symbol => String | nil}]
|
16
|
+
def to_h
|
17
|
+
# Rely on Dry::Struct's default to_h, filtering out nil label.
|
18
|
+
super.compact
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a branch in a Gitgraph diagram.
|
6
|
+
class GitBranch < Dry::Struct
|
7
|
+
include Elements::Types
|
8
|
+
|
9
|
+
attribute :name, Types::Strict::String.constrained(min_size: 1)
|
10
|
+
# head_commit_id can be nil initially if the branch is created before any commits
|
11
|
+
attribute :head_commit_id, Types::Strict::String.optional.default(nil)
|
12
|
+
attribute :start_commit_id, Types::Strict::String.constrained(min_size: 1)
|
13
|
+
|
14
|
+
# Returns a hash representation suitable for serialization.
|
15
|
+
#
|
16
|
+
# @return [Hash{Symbol => String}]
|
17
|
+
def to_h
|
18
|
+
hash = {
|
19
|
+
name:,
|
20
|
+
start_commit_id:
|
21
|
+
}
|
22
|
+
hash[:head_commit_id] = head_commit_id if head_commit_id
|
23
|
+
hash
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -0,0 +1,36 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a commit in a Gitgraph diagram.
|
6
|
+
class GitCommit < Dry::Struct
|
7
|
+
include Elements::Types
|
8
|
+
|
9
|
+
attribute :id, Types::Strict::String.constrained(min_size: 1)
|
10
|
+
attribute :parent_ids, Types::Strict::Array.of(Types::Strict::String).default([].freeze)
|
11
|
+
attribute :branch_name, Types::Strict::String.constrained(min_size: 1)
|
12
|
+
attribute :message, Types::Strict::String.optional.default(nil)
|
13
|
+
attribute :tag, Types::Strict::String.optional.default(nil)
|
14
|
+
attribute :type, Types::Strict::Symbol.default(:NORMAL).enum(:NORMAL, :REVERSE, :HIGHLIGHT, :MERGE, :CHERRY_PICK)
|
15
|
+
attribute :cherry_pick_source_id, Types::Strict::String.optional.default(nil)
|
16
|
+
|
17
|
+
# Returns a hash representation suitable for serialization.
|
18
|
+
# Dry::Struct provides to_h, but explicit definition ensures desired keys/structure.
|
19
|
+
# Optional attributes are included only if they have non-nil values.
|
20
|
+
#
|
21
|
+
# @return [Hash{Symbol => String | Array<String> | Symbol}]
|
22
|
+
def to_h
|
23
|
+
hash = {
|
24
|
+
id:,
|
25
|
+
parent_ids:,
|
26
|
+
branch_name:,
|
27
|
+
type:
|
28
|
+
}
|
29
|
+
hash[:message] = message if message
|
30
|
+
hash[:tag] = tag if tag
|
31
|
+
hash[:cherry_pick_source_id] = cherry_pick_source_id if cherry_pick_source_id
|
32
|
+
hash
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
# Namespace for diagram element value objects.
|
5
|
+
|
6
|
+
module Elements
|
7
|
+
# Represents a node in various diagram types (e.g., Flowchart).
|
8
|
+
# Typically has an identifier and a display label.
|
9
|
+
class Node < Dry::Struct
|
10
|
+
# Use the shared Types module
|
11
|
+
include Elements::Types
|
12
|
+
|
13
|
+
attribute :id, Types::Strict::String.constrained(min_size: 1)
|
14
|
+
attribute :label, Types::Strict::String.constrained(min_size: 1)
|
15
|
+
|
16
|
+
# Returns a hash representation suitable for serialization.
|
17
|
+
#
|
18
|
+
# @return [Hash{Symbol => String}]
|
19
|
+
def to_h
|
20
|
+
{
|
21
|
+
id:,
|
22
|
+
label:
|
23
|
+
}
|
24
|
+
# Dry::Struct automatically provides a to_h method,
|
25
|
+
# but defining it explicitly ensures the desired structure.
|
26
|
+
# super # Alternatively, call super if the default is sufficient.
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
@@ -0,0 +1,33 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a relationship (e.g., association, inheritance) between two classes
|
6
|
+
# in a UML Class Diagram.
|
7
|
+
class Relationship < Dry::Struct
|
8
|
+
# Use the shared Types module
|
9
|
+
include Elements::Types
|
10
|
+
|
11
|
+
# Name of the source class
|
12
|
+
attribute :source_class_name, Types::Strict::String.constrained(min_size: 1)
|
13
|
+
|
14
|
+
# Name of the target class
|
15
|
+
attribute :target_class_name, Types::Strict::String.constrained(min_size: 1)
|
16
|
+
|
17
|
+
# Type of relationship (e.g., "association", "inheritance", "composition")
|
18
|
+
# Consider using a constrained string or enum type later if needed.
|
19
|
+
attribute :type, Types::Strict::String.constrained(min_size: 1)
|
20
|
+
|
21
|
+
# Optional label for the relationship (e.g., multiplicity, role name)
|
22
|
+
attribute :label, Types::Strict::String.optional.default(nil)
|
23
|
+
|
24
|
+
# Returns a hash representation suitable for serialization.
|
25
|
+
#
|
26
|
+
# @return [Hash{Symbol => String | nil}]
|
27
|
+
def to_h
|
28
|
+
# Rely on Dry::Struct's default to_h, filtering out nil label.
|
29
|
+
super.compact
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a slice in a Pie Diagram.
|
6
|
+
class Slice < Dry::Struct
|
7
|
+
# Use the shared Types module
|
8
|
+
include Elements::Types
|
9
|
+
|
10
|
+
attribute :label, Types::Strict::String.constrained(min_size: 1)
|
11
|
+
# Represents the raw value of the slice (not percentage)
|
12
|
+
attribute :value, Types::Coercible::Float.constrained(gteq: 0)
|
13
|
+
# Calculated percentage (read-only)
|
14
|
+
attribute :percentage, Types::Strict::Float.optional.default(nil).meta(reader: true)
|
15
|
+
# Consider adding optional color attribute later.
|
16
|
+
# attribute :color, Types::Strict::String.optional.default(nil)
|
17
|
+
|
18
|
+
# Returns a hash representation suitable for serialization.
|
19
|
+
#
|
20
|
+
# @return [Hash{Symbol => String | Float | nil}]
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a state in a State Diagram.
|
6
|
+
class State < Dry::Struct
|
7
|
+
# Use the shared Types module
|
8
|
+
include Elements::Types
|
9
|
+
|
10
|
+
attribute :id, Types::Strict::String.constrained(min_size: 1)
|
11
|
+
attribute :label, Types::Strict::String.optional.default(nil)
|
12
|
+
# TODO: Consider adding back type attribute (e.g., start, end, state)
|
13
|
+
# attribute :type, Types::Strict::String.enum('state', 'start', 'end', 'fork', 'join').default('state')
|
14
|
+
|
15
|
+
# Returns a hash representation suitable for serialization.
|
16
|
+
#
|
17
|
+
# @return [Hash{Symbol => String | nil}]
|
18
|
+
def to_h
|
19
|
+
# Rely on Dry::Struct's default to_h, filtering out nil label.
|
20
|
+
super.compact
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Diagrams
|
4
|
+
module Elements
|
5
|
+
# Represents a task in a Gantt Diagram.
|
6
|
+
class Task < Dry::Struct
|
7
|
+
# Use the shared Types module
|
8
|
+
include Elements::Types
|
9
|
+
|
10
|
+
attribute :id, Types::Strict::String.constrained(min_size: 1)
|
11
|
+
attribute :name, Types::Strict::String.constrained(min_size: 1)
|
12
|
+
# Using String for dates initially for simplicity.
|
13
|
+
# Consider Types::Strict::Date or custom coercible types later.
|
14
|
+
attribute :start_date, Types::Strict::String.constrained(min_size: 1) # Basic check
|
15
|
+
attribute :end_date, Types::Strict::String.constrained(min_size: 1) # Basic check
|
16
|
+
# TODO: Add dependencies attribute (e.g., Types::Strict::Array.of(Types::Strict::String))
|
17
|
+
|
18
|
+
# Returns a hash representation suitable for serialization.
|
19
|
+
#
|
20
|
+
# @return [Hash{Symbol => String}]
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|