diagram 0.3.0 → 0.3.3

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 (47) hide show
  1. checksums.yaml +4 -4
  2. data/lib/diagram.rb +3 -0
  3. data/lib/diagrams/base.rb +8 -6
  4. data/lib/diagrams/class_diagram.rb +0 -4
  5. data/lib/diagrams/elements/class_entity.rb +0 -4
  6. data/lib/diagrams/elements/edge.rb +0 -4
  7. data/lib/diagrams/elements/erd_attribute.rb +28 -0
  8. data/lib/diagrams/elements/erd_entity.rb +23 -0
  9. data/lib/diagrams/elements/erd_relationship.rb +40 -0
  10. data/lib/diagrams/elements/event.rb +0 -4
  11. data/lib/diagrams/elements/gantt_section.rb +23 -0
  12. data/lib/diagrams/elements/git_branch.rb +27 -0
  13. data/lib/diagrams/elements/git_commit.rb +36 -0
  14. data/lib/diagrams/elements/node.rb +0 -8
  15. data/lib/diagrams/elements/relationship.rb +0 -4
  16. data/lib/diagrams/elements/slice.rb +0 -4
  17. data/lib/diagrams/elements/state.rb +0 -4
  18. data/lib/diagrams/elements/task.rb +20 -12
  19. data/lib/diagrams/elements/timeline_event.rb +21 -0
  20. data/lib/diagrams/elements/timeline_period.rb +23 -0
  21. data/lib/diagrams/elements/timeline_section.rb +23 -0
  22. data/lib/diagrams/elements/transition.rb +0 -4
  23. data/lib/diagrams/elements.rb +12 -0
  24. data/lib/diagrams/er_diagram.rb +143 -0
  25. data/lib/diagrams/flowchart_diagram.rb +0 -6
  26. data/lib/diagrams/gantt_diagram.rb +118 -35
  27. data/lib/diagrams/gitgraph_diagram.rb +345 -0
  28. data/lib/diagrams/pie_diagram.rb +0 -3
  29. data/lib/diagrams/state_diagram.rb +0 -5
  30. data/lib/diagrams/timeline_diagram.rb +159 -0
  31. data/lib/diagrams/version.rb +1 -1
  32. data/lib/diagrams.rb +13 -1
  33. data/sig/diagrams/elements/erd_attribute.rbs +20 -0
  34. data/sig/diagrams/elements/erd_entity.rbs +20 -0
  35. data/sig/diagrams/elements/erd_relationship.rbs +25 -0
  36. data/sig/diagrams/elements/gantt_section.rbs +18 -0
  37. data/sig/diagrams/elements/git_branch.rbs +19 -0
  38. data/sig/diagrams/elements/git_commit.rbs +23 -0
  39. data/sig/diagrams/elements/task.rbs +13 -5
  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/er_diagram.rbs +30 -0
  44. data/sig/diagrams/gantt_diagram.rbs +28 -10
  45. data/sig/diagrams/gitgraph_diagram.rbs +35 -0
  46. data/sig/diagrams/timeline_diagram.rbs +14 -0
  47. metadata +28 -2
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: b4af5a1be5db0b92fbdd6a3c6d41eec3d78dd3bd36c6ba6bb70897f552bfdf27
4
- data.tar.gz: 967838b7e5d58ee6ffa188cfc623a130fdb0223e586118c3e11ea1c66fff16cc
3
+ metadata.gz: c99defb74ddced6d50a0dd2bd5cf61b90ce2edab9192c5f119586442eb625d12
4
+ data.tar.gz: 1bd666eed832e6052f789c3a37fd634d2cb4c198468281bafc17bc1ca500d05a
5
5
  SHA512:
6
- metadata.gz: 90b0513c6e57596f3c769f01a693b1615cbd9044f8dfc2d97a17432073b9f5415b23b0724c91f121219b767f892d4977eac598e33159d4ca83cd892bf00235f3
7
- data.tar.gz: 1dac456dbecc729f32da2a863bd0605e9ee4b830982920b56c54b68dae4ce27ad635e0ad68bd6fc5029838a5f994553da9592dd09390301f465b1f4ceea09ca9
6
+ metadata.gz: f9d0b5269c5093821a09140273590af586dc35e4a21b9aed5c45a5201db8ebcfdb769e3764038fe0a74e467eea3d18e1f1fc970c295e69fefbdc686230106b48
7
+ data.tar.gz: 7a53a67aecab0fa28adee1e6f826114ad0f0184e994e31c346d9b3224a4fd72e255ed4eaea714327c467664cef766ae2da1cbded21c58d620d7cb5862c65617d
data/lib/diagram.rb ADDED
@@ -0,0 +1,3 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'diagrams'
data/lib/diagrams/base.rb CHANGED
@@ -1,9 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'digest'
4
- require 'json'
5
- require 'dry-equalizer'
6
-
7
3
  module Diagrams
8
4
  # Abstract base class for all diagram types.
9
5
  # Provides common functionality like versioning, checksum calculation,
@@ -69,12 +65,14 @@ module Diagrams
69
65
  self_collection = self_elements[type] || []
70
66
  other_collection = other_elements[type] || []
71
67
 
72
- # Determine identifier method (prefer id, then name, fallback to object itself)
68
+ # Determine identifier method (prefer id, then name, then title, then label, fallback to object itself)
73
69
  identifier_method = if self_collection.first.respond_to?(:id)
74
70
  :id
75
71
  elsif self_collection.first.respond_to?(:name)
76
72
  :name
77
- elsif self_collection.first.respond_to?(:label) # For Slice
73
+ elsif self_collection.first.respond_to?(:title) # For TimelineSection
74
+ :title
75
+ elsif self_collection.first.respond_to?(:label) # For Slice, TimelinePeriod
78
76
  :label
79
77
  else
80
78
  :itself # Fallback to object identity/equality
@@ -198,6 +196,10 @@ module Diagrams
198
196
  # Simple helper to convert snake_case to CamelCase
199
197
  # (Avoids ActiveSupport dependency)
200
198
  def snake_to_camel_case(string)
199
+ # Handle specific acronyms first
200
+ return 'ERDiagram' if string == 'er_diagram'
201
+
202
+ # Default conversion
201
203
  string.split('_').collect(&:capitalize).join
202
204
  end
203
205
  end
@@ -1,9 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/class_entity'
5
- require_relative 'elements/relationship'
6
-
7
3
  module Diagrams
8
4
  # Represents a UML Class Diagram consisting of classes and relationships between them.
9
5
  class ClassDiagram < Base
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a class entity in a UML Class Diagram.
10
6
  class ClassEntity < Dry::Struct
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents an edge or link between two nodes in a diagram.
10
6
  # Typically connects nodes via their IDs and can have an optional label.
@@ -0,0 +1,28 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents an attribute within an ERD entity.
6
+ class ERDAttribute < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :type, Types::Strict::String.constrained(min_size: 1)
10
+ attribute :name, Types::Strict::String.constrained(min_size: 1)
11
+ attribute :keys, Types::Strict::Array.of(Types::Strict::Symbol.enum(:PK, :FK, :UK)).default([].freeze)
12
+ attribute :comment, Types::Strict::String.optional.default(nil)
13
+
14
+ # Returns a hash representation suitable for serialization.
15
+ #
16
+ # @return [Hash{Symbol => String | Array<Symbol>}]
17
+ def to_h
18
+ hash = {
19
+ type:,
20
+ name:,
21
+ keys: keys.map(&:to_s) # Convert symbols to strings for serialization
22
+ }
23
+ hash[:comment] = comment if comment
24
+ hash
25
+ end
26
+ end
27
+ end
28
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents an entity (table) in an ER Diagram.
6
+ class ERDEntity < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :name, Types::Strict::String.constrained(min_size: 1)
10
+ attribute :entity_attributes, Types::Strict::Array.of(ERDAttribute).default([].freeze)
11
+
12
+ # Returns a hash representation suitable for serialization.
13
+ #
14
+ # @return [Hash{Symbol => String | Array<Hash>}]
15
+ def to_h
16
+ {
17
+ name: name,
18
+ attributes: entity_attributes.map(&:to_h) # Renamed variable
19
+ }
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,40 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a relationship between two entities in an ER Diagram.
6
+ class ERDRelationship < Dry::Struct
7
+ include Elements::Types
8
+
9
+ # Cardinality symbols (Crow's Foot notation mapping)
10
+ CARDINALITY = Types::Strict::Symbol.enum(
11
+ :ZERO_OR_ONE, # |o
12
+ :ONE_ONLY, # ||
13
+ :ZERO_OR_MORE, # }o
14
+ :ONE_OR_MORE # }|
15
+ )
16
+
17
+ attribute :entity1, Types::Strict::String.constrained(min_size: 1)
18
+ attribute :entity2, Types::Strict::String.constrained(min_size: 1)
19
+ attribute :cardinality1, CARDINALITY # Cardinality of entity1 relative to entity2
20
+ attribute :cardinality2, CARDINALITY # Cardinality of entity2 relative to entity1
21
+ attribute :identifying, Types::Strict::Bool.default(false) # Is it an identifying relationship? (solid vs dashed line)
22
+ attribute :label, Types::Strict::String.optional.default(nil) # Optional action/verb phrase
23
+
24
+ # Returns a hash representation suitable for serialization.
25
+ #
26
+ # @return [Hash{Symbol => String | Symbol | Bool}]
27
+ def to_h
28
+ hash = {
29
+ entity1: entity1,
30
+ entity2: entity2,
31
+ cardinality1: cardinality1.to_s, # Convert symbol to string
32
+ cardinality2: cardinality2.to_s, # Convert symbol to string
33
+ identifying: identifying
34
+ }
35
+ hash[:label] = label if label
36
+ hash
37
+ end
38
+ end
39
+ end
40
+ end
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents an event, potentially used in State Diagrams or others.
10
6
  class Event < Dry::Struct
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a section within a Gantt chart, grouping multiple tasks.
6
+ class GanttSection < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :title, Types::Strict::String.constrained(min_size: 1)
10
+ attribute :tasks, Types::Strict::Array.of(Task).default([].freeze)
11
+
12
+ # Returns a hash representation suitable for serialization.
13
+ #
14
+ # @return [Hash{Symbol => String | Array<Hash>}]
15
+ def to_h
16
+ {
17
+ title:,
18
+ tasks: tasks.map(&:to_h)
19
+ }
20
+ end
21
+ end
22
+ end
23
+ 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
@@ -1,17 +1,9 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require 'dry-types'
5
-
6
3
  module Diagrams
7
4
  # Namespace for diagram element value objects.
8
5
 
9
6
  module Elements
10
- # Common Dry::Types definitions for elements.
11
- module Types
12
- include Dry.Types()
13
- end
14
-
15
7
  # Represents a node in various diagram types (e.g., Flowchart).
16
8
  # Typically has an identifier and a display label.
17
9
  class Node < Dry::Struct
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a relationship (e.g., association, inheritance) between two classes
10
6
  # in a UML Class Diagram.
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a slice in a Pie Diagram.
10
6
  class Slice < Dry::Struct
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a state in a State Diagram.
10
6
  class State < Dry::Struct
@@ -1,27 +1,35 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a task in a Gantt Diagram.
10
6
  class Task < Dry::Struct
11
7
  # Use the shared Types module
12
8
  include Elements::Types
13
9
 
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))
10
+ # Status symbols allowed by Mermaid Gantt
11
+ STATUS = Types::Strict::Symbol.enum(:done, :active, :crit)
12
+
13
+ # Attributes
14
+ attribute :id, Types::Strict::String.constrained(min_size: 1) # Unique ID for dependencies
15
+ attribute :label, Types::Strict::String.constrained(min_size: 1) # Display name
16
+ attribute :status, STATUS.optional.default(nil) # Task status (nil implies default/future)
17
+ attribute :start, Types::Strict::String.constrained(min_size: 1) # Start date, task ID, or 'after taskX[, taskY]'
18
+ attribute :duration, Types::Strict::String.constrained(min_size: 1) # Duration string (e.g., '7d', '2w')
21
19
 
22
20
  # Returns a hash representation suitable for serialization.
23
21
  #
24
- # @return [Hash{Symbol => String}]
22
+ # @return [Hash{Symbol => String | Symbol | nil}]
23
+ def to_h
24
+ hash = {
25
+ id: id,
26
+ label: label,
27
+ start: start,
28
+ duration: duration
29
+ }
30
+ hash[:status] = status if status # Include status only if set
31
+ hash
32
+ end
25
33
  end
26
34
  end
27
35
  end
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a single event description within a timeline period.
6
+ class TimelineEvent < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :description, Types::Strict::String.constrained(min_size: 1)
10
+
11
+ # Returns a hash representation suitable for serialization.
12
+ #
13
+ # @return [Hash{Symbol => String}]
14
+ def to_h
15
+ {
16
+ description:
17
+ }
18
+ end
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a specific time period on the timeline, containing one or more events.
6
+ class TimelinePeriod < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :label, Types::Strict::String.constrained(min_size: 1)
10
+ attribute :events, Types::Strict::Array.of(TimelineEvent).constrained(min_size: 1)
11
+
12
+ # Returns a hash representation suitable for serialization.
13
+ #
14
+ # @return [Hash{Symbol => String | Array<Hash>}]
15
+ def to_h
16
+ {
17
+ label:,
18
+ events: events.map(&:to_h)
19
+ }
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a section or age within the timeline, grouping multiple time periods.
6
+ class TimelineSection < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :title, Types::Strict::String.constrained(min_size: 1)
10
+ attribute :periods, Types::Strict::Array.of(TimelinePeriod).default([].freeze)
11
+
12
+ # Returns a hash representation suitable for serialization.
13
+ #
14
+ # @return [Hash{Symbol => String | Array<Hash>}]
15
+ def to_h
16
+ {
17
+ title:,
18
+ periods: periods.map(&:to_h)
19
+ }
20
+ end
21
+ end
22
+ end
23
+ end
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a transition between two states in a State Diagram.
10
6
  class Transition < Dry::Struct
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ # Namespace for diagram element value objects.
5
+
6
+ module Elements
7
+ # Common Dry::Types definitions for elements.
8
+ module Types
9
+ include Dry.Types()
10
+ end
11
+ end
12
+ end
@@ -0,0 +1,143 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ # Represents an Entity Relationship Diagram (ERD).
5
+ class ERDiagram < Base
6
+ attr_reader :entities, :relationships
7
+
8
+ # Initializes a new ERDiagram.
9
+ #
10
+ # @param entities [Array<Element::ERDEntity>] Initial entities (optional).
11
+ # @param relationships [Array<Element::ERDRelationship>] Initial relationships (optional).
12
+ # @param version [String, Integer, nil] User-defined version identifier.
13
+ def initialize(entities: [], relationships: [], version: 1)
14
+ super(version:)
15
+ @entities = (entities.is_a?(Array) ? entities : []).each_with_object({}) { |e, h| h[e.name] = e }
16
+ @relationships = relationships.is_a?(Array) ? relationships : []
17
+ validate_relationships!
18
+ update_checksum!
19
+ end
20
+
21
+ # Adds an entity to the diagram.
22
+ #
23
+ # @param name [String] The unique name of the entity.
24
+ # @param attributes [Array<Hash>] Array of attribute definitions (hashes like { type:, name:, keys:, comment: }).
25
+ # @raise [ArgumentError] if an entity with the same name already exists.
26
+ # @return [Elements::ERDEntity] The added entity.
27
+ def add_entity(name:, attributes: [])
28
+ raise ArgumentError, "Entity name '#{name}' cannot be empty" if name.nil? || name.strip.empty?
29
+ raise ArgumentError, "Entity with name '#{name}' already exists" if @entities.key?(name)
30
+
31
+ entity_attributes = attributes.map do |attr_hash|
32
+ Elements::ERDAttribute.new(attr_hash.transform_keys(&:to_sym))
33
+ end
34
+ new_entity = Elements::ERDEntity.new(name:, entity_attributes: entity_attributes) # Use renamed attribute
35
+
36
+ @entities[name] = new_entity
37
+ update_checksum!
38
+ new_entity
39
+ end
40
+
41
+ # Adds a relationship between two entities.
42
+ #
43
+ # @param entity1 [String] Name of the first entity.
44
+ # @param entity2 [String] Name of the second entity.
45
+ # @param cardinality1 [Symbol] Cardinality of entity1 relative to entity2 (e.g., :ONE_ONLY).
46
+ # @param cardinality2 [Symbol] Cardinality of entity2 relative to entity1 (e.g., :ZERO_OR_MORE).
47
+ # @param identifying [Boolean] Whether the relationship is identifying (default: false).
48
+ # @param label [String, nil] Optional label describing the relationship action.
49
+ # @raise [ArgumentError] if either entity does not exist.
50
+ # @return [Elements::ERDRelationship] The added relationship.
51
+ def add_relationship(entity1:, entity2:, cardinality1:, cardinality2:, identifying: false, label: nil)
52
+ unless @entities.key?(entity1) && @entities.key?(entity2)
53
+ raise ArgumentError, "One or both entities ('#{entity1}', '#{entity2}') not found for relationship."
54
+ end
55
+
56
+ new_relationship = Elements::ERDRelationship.new(
57
+ entity1:,
58
+ entity2:,
59
+ cardinality1:,
60
+ cardinality2:,
61
+ identifying:,
62
+ label:
63
+ )
64
+ @relationships << new_relationship
65
+ update_checksum!
66
+ new_relationship
67
+ end
68
+
69
+ # Finds an entity by its name.
70
+ #
71
+ # @param entity_name [String] The name of the entity to find.
72
+ # @return [Elements::ERDEntity, nil] The found entity or nil.
73
+ def find_entity(entity_name)
74
+ @entities[entity_name]
75
+ end
76
+
77
+ # --- Base Class Implementation ---
78
+
79
+ def to_h_content
80
+ {
81
+ entities: @entities.values.map(&:to_h),
82
+ relationships: @relationships.map(&:to_h)
83
+ }
84
+ end
85
+
86
+ def identifiable_elements
87
+ {
88
+ entities: @entities.values,
89
+ relationships: @relationships # Relationships don't have a simple unique ID, rely on object equality for diff
90
+ }
91
+ end
92
+
93
+ def self.from_h(data_hash, version:, checksum:)
94
+ entities_data = data_hash[:entities] || data_hash['entities'] || []
95
+ relationships_data = data_hash[:relationships] || data_hash['relationships'] || []
96
+
97
+ entities = entities_data.map do |entity_h|
98
+ entity_data = entity_h.transform_keys(&:to_sym)
99
+ attributes_data = entity_data[:entity_attributes] || entity_data[:attributes] || [] # Accept both old and new key for now
100
+ attributes = attributes_data.map do |attr_h|
101
+ attr_data = attr_h.transform_keys(&:to_sym)
102
+ # Convert keys back to symbols if they are strings
103
+ # Convert keys back to symbols
104
+ attr_data[:keys] = attr_data[:keys].map(&:to_sym) if attr_data[:keys].is_a?(Array)
105
+ Elements::ERDAttribute.new(attr_data)
106
+ end
107
+ # Use the correct attribute name when creating the entity
108
+ Elements::ERDEntity.new(entity_data.merge(entity_attributes: attributes))
109
+ end
110
+
111
+ relationships = relationships_data.map do |rel_h|
112
+ rel_data = rel_h.transform_keys(&:to_sym)
113
+ # Convert cardinalities back to symbols if they are strings
114
+ rel_data[:cardinality1] = rel_data[:cardinality1].to_sym if rel_data[:cardinality1].is_a?(String)
115
+ rel_data[:cardinality2] = rel_data[:cardinality2].to_sym if rel_data[:cardinality2].is_a?(String)
116
+ Elements::ERDRelationship.new(rel_data)
117
+ end
118
+
119
+ diagram = new(entities:, relationships:, version:)
120
+
121
+ # Optional: Verify checksum
122
+ if checksum && diagram.checksum != checksum
123
+ warn "Checksum mismatch for loaded ERDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
124
+ end
125
+
126
+ diagram
127
+ end
128
+
129
+ private
130
+
131
+ # Validates that all relationships refer to existing entities during initialization.
132
+ def validate_relationships!
133
+ @relationships.each do |rel|
134
+ unless @entities.key?(rel.entity1) && @entities.key?(rel.entity2)
135
+ raise ArgumentError, "Relationship refers to non-existent entity IDs ('#{rel.entity1}' or '#{rel.entity2}')"
136
+ end
137
+ end
138
+ end
139
+
140
+ # Protected method access
141
+ protected :update_checksum!
142
+ end
143
+ end
@@ -1,9 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/node'
5
- require_relative 'elements/edge'
6
-
7
3
  module Diagrams
8
4
  # Represents a flowchart diagram consisting of nodes and edges connecting them.
9
5
  class FlowchartDiagram < Base
@@ -28,7 +24,6 @@ module Diagrams
28
24
  # @raise [ArgumentError] if a node with the same ID already exists.
29
25
  # @return [Element::Node] The added node.
30
26
  def add_node(node)
31
-
32
27
  raise ArgumentError, 'Node must be a Diagrams::Element::Node' unless node.is_a?(Diagrams::Elements::Node)
33
28
  raise ArgumentError, "Node with ID '#{node.id}' already exists" if find_node(node.id)
34
29
 
@@ -43,7 +38,6 @@ module Diagrams
43
38
  # @raise [ArgumentError] if the edge refers to non-existent node IDs.
44
39
  # @return [Element::Edge] The added edge.
45
40
  def add_edge(edge)
46
-
47
41
  raise ArgumentError, 'Edge must be a Diagrams::Element::Edge' unless edge.is_a?(Diagrams::Elements::Edge)
48
42
  unless find_node(edge.source_id) && find_node(edge.target_id)
49
43
  raise ArgumentError, "Edge refers to non-existent node IDs ('#{edge.source_id}' or '#{edge.target_id}')"