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.
- checksums.yaml +4 -4
- data/lib/diagram.rb +3 -0
- data/lib/diagrams/base.rb +8 -6
- data/lib/diagrams/class_diagram.rb +0 -4
- data/lib/diagrams/elements/class_entity.rb +0 -4
- data/lib/diagrams/elements/edge.rb +0 -4
- data/lib/diagrams/elements/erd_attribute.rb +28 -0
- data/lib/diagrams/elements/erd_entity.rb +23 -0
- data/lib/diagrams/elements/erd_relationship.rb +40 -0
- data/lib/diagrams/elements/event.rb +0 -4
- data/lib/diagrams/elements/gantt_section.rb +23 -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 +0 -8
- data/lib/diagrams/elements/relationship.rb +0 -4
- data/lib/diagrams/elements/slice.rb +0 -4
- data/lib/diagrams/elements/state.rb +0 -4
- data/lib/diagrams/elements/task.rb +20 -12
- 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 +0 -4
- data/lib/diagrams/elements.rb +12 -0
- data/lib/diagrams/er_diagram.rb +143 -0
- data/lib/diagrams/flowchart_diagram.rb +0 -6
- data/lib/diagrams/gantt_diagram.rb +118 -35
- data/lib/diagrams/gitgraph_diagram.rb +345 -0
- data/lib/diagrams/pie_diagram.rb +0 -3
- data/lib/diagrams/state_diagram.rb +0 -5
- data/lib/diagrams/timeline_diagram.rb +159 -0
- data/lib/diagrams/version.rb +1 -1
- data/lib/diagrams.rb +13 -1
- data/sig/diagrams/elements/erd_attribute.rbs +20 -0
- data/sig/diagrams/elements/erd_entity.rbs +20 -0
- data/sig/diagrams/elements/erd_relationship.rbs +25 -0
- data/sig/diagrams/elements/gantt_section.rbs +18 -0
- data/sig/diagrams/elements/git_branch.rbs +19 -0
- data/sig/diagrams/elements/git_commit.rbs +23 -0
- data/sig/diagrams/elements/task.rbs +13 -5
- 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/er_diagram.rbs +30 -0
- data/sig/diagrams/gantt_diagram.rbs +28 -10
- data/sig/diagrams/gitgraph_diagram.rbs +35 -0
- data/sig/diagrams/timeline_diagram.rbs +14 -0
- metadata +28 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: c99defb74ddced6d50a0dd2bd5cf61b90ce2edab9192c5f119586442eb625d12
|
4
|
+
data.tar.gz: 1bd666eed832e6052f789c3a37fd634d2cb4c198468281bafc17bc1ca500d05a
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: f9d0b5269c5093821a09140273590af586dc35e4a21b9aed5c45a5201db8ebcfdb769e3764038fe0a74e467eea3d18e1f1fc970c295e69fefbdc686230106b48
|
7
|
+
data.tar.gz: 7a53a67aecab0fa28adee1e6f826114ad0f0184e994e31c346d9b3224a4fd72e255ed4eaea714327c467664cef766ae2da1cbded21c58d620d7cb5862c65617d
|
data/lib/diagram.rb
ADDED
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?(:
|
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 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
|
@@ -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,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
|
-
|
15
|
-
|
16
|
-
|
17
|
-
#
|
18
|
-
attribute :
|
19
|
-
attribute :
|
20
|
-
|
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
|
@@ -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}')"
|