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
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 6ef3c343a2636e4f7ffda61678ff784b5d0b083d25ab83364e0285cb5c150acd
4
- data.tar.gz: 9988256feb04b4eee350d792cc697cdb3bee87371a0211f5f714a4f1c380d2e8
3
+ metadata.gz: b6a7329e68d37cab7fe076fb763b56fddff42b23e81619bd70f3a89ef82ebff8
4
+ data.tar.gz: 3eac3513c7ebb76307967f12f3c9d189e35022203ce817e3fbc90be310ac7b10
5
5
  SHA512:
6
- metadata.gz: b9bf6af4e963b22981ce1eec3205e1d35e3a25752183fc2b91b1650148e3eb08f738522535b6234eca33cab21be7070132f8f41f4dff9663fa35bff8cf1d742d
7
- data.tar.gz: 4ce8d1844a7bd0c961a89ed838d3541cbf10a9794f897622aea05f81b654937acc9054c7a09a114d625e22bfe12af524ea7832e3a20a1f40f0ca1e9c39548fb1
6
+ metadata.gz: 0b59a02a496318c080f8ff72bbfb6bb224ae4dfeb3657eda3e262ac401cca6871f8a18a487fbe012699c7016ae8ef07cfc565f224e7697cc3a6aba9abb60907b
7
+ data.tar.gz: 6cee2aa87b44ae4aab58a99fcf9cfadfbde2f9496523d7e02e5dcb00f6c4ec14abce8b36ff79c1b276e44eec407fe09db6dcaabd9a14c83a050b7684df854f41
data/lib/diagram.rb ADDED
@@ -0,0 +1,3 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'diagrams'
@@ -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
- class ClassDiagram < AbstractDiagram
5
- attribute :classes, ClassDiagram::Types::Array.of(Class)
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