diagram 0.2.1 → 0.3.0

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