table_sync 6.8.0 → 6.9.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +12 -0
- data/Gemfile.lock +1 -1
- data/lib/table_sync/receiving/handler.rb +8 -0
- data/lib/table_sync/receiving/model/active_record.rb +6 -1
- data/lib/table_sync/receiving/model/sequel.rb +6 -1
- data/lib/table_sync/utils/schema/builder/active_record.rb +43 -0
- data/lib/table_sync/utils/schema/builder/sequel.rb +37 -0
- data/lib/table_sync/utils/schema/builder.rb +5 -0
- data/lib/table_sync/utils/schema/validator/type.rb +61 -0
- data/lib/table_sync/utils/schema/validator.rb +21 -0
- data/lib/table_sync/utils/schema.rb +28 -0
- data/lib/table_sync/utils.rb +1 -0
- data/lib/table_sync/version.rb +1 -1
- metadata +9 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: '09c6fd6d3ac06db07e1ac54edd3b5dbbcfb96b15191450f1fc71750087ecb660'
|
4
|
+
data.tar.gz: 4270f9f7a0037128664b9ee48b57ddf349304e46654a39a1cec08f59036d10da
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 95b363fd2d16fae609205940176ff31069864ee044ce691fbf6fbde5e5e5153c53e8bc1fa5089e42a32b310014f4970ab8efa35bb972b68f4734e75d5978a009
|
7
|
+
data.tar.gz: d1ef68707a2ec0cee63edd72a55511c8107fe5b056458cd808444a46d1fbfd8d1f03dbcb394901d269389029b0f9438034f7a36c4058eb95c3756c7a9ba1d17d
|
data/CHANGELOG.md
CHANGED
@@ -1,6 +1,18 @@
|
|
1
1
|
# Changelog
|
2
2
|
All notable changes to this project will be documented in this file.
|
3
3
|
|
4
|
+
## [6.9.2] - 2025-10-03
|
5
|
+
### Fixed
|
6
|
+
- Fix of all types taking type cast into account
|
7
|
+
|
8
|
+
## [6.9.1] - 2025-10-02
|
9
|
+
### Fixed
|
10
|
+
- DateTime can be String now
|
11
|
+
|
12
|
+
## [6.9.0] - 2025-10-02
|
13
|
+
### Added
|
14
|
+
- Add check of data scalar types in order to provide readable error text
|
15
|
+
|
4
16
|
## [6.8.0] - 2025-08-25
|
5
17
|
### Changed
|
6
18
|
- Updated `rabbit_messaging` gem version to `1.7.0`
|
data/Gemfile.lock
CHANGED
@@ -131,12 +131,20 @@ class TableSync::Receiving::Handler < Rabbit::EventHandler
|
|
131
131
|
end
|
132
132
|
end
|
133
133
|
|
134
|
+
def validate_data_types(model, data)
|
135
|
+
errors = model.validate_types(data)
|
136
|
+
return if errors.nil?
|
137
|
+
|
138
|
+
raise TableSync::DataError.new(data, errors.keys, errors.to_json)
|
139
|
+
end
|
140
|
+
|
134
141
|
def perform(config, params)
|
135
142
|
model = config.model
|
136
143
|
|
137
144
|
model.transaction do
|
138
145
|
results = if event == :update
|
139
146
|
config.before_update(**params)
|
147
|
+
validate_data_types(model, params[:data])
|
140
148
|
model.upsert(**params)
|
141
149
|
else
|
142
150
|
config.before_destroy(**params)
|
@@ -25,6 +25,7 @@ module TableSync::Receiving::Model
|
|
25
25
|
self.table_name = table_name
|
26
26
|
self.inheritance_column = nil
|
27
27
|
end
|
28
|
+
@types_validator = TableSync::Utils::Schema::Builder::ActiveRecord.build(@raw_model)
|
28
29
|
|
29
30
|
model_naming = ::TableSync::NamingResolver::ActiveRecord.new(table_name:)
|
30
31
|
|
@@ -105,6 +106,10 @@ module TableSync::Receiving::Model
|
|
105
106
|
result
|
106
107
|
end
|
107
108
|
|
109
|
+
def validate_types(data)
|
110
|
+
types_validator.validate(data)
|
111
|
+
end
|
112
|
+
|
108
113
|
def transaction(&)
|
109
114
|
::ActiveRecord::Base.transaction(&)
|
110
115
|
end
|
@@ -115,7 +120,7 @@ module TableSync::Receiving::Model
|
|
115
120
|
|
116
121
|
private
|
117
122
|
|
118
|
-
attr_reader :raw_model
|
123
|
+
attr_reader :raw_model, :types_validator
|
119
124
|
|
120
125
|
def db
|
121
126
|
raw_model.connection
|
@@ -6,6 +6,7 @@ module TableSync::Receiving::Model
|
|
6
6
|
|
7
7
|
def initialize(table_name)
|
8
8
|
@raw_model = Class.new(::Sequel::Model(table_name)).tap(&:unrestrict_primary_key)
|
9
|
+
@types_validator = TableSync::Utils::Schema::Builder::Sequel.build(@raw_model)
|
9
10
|
|
10
11
|
model_naming = ::TableSync::NamingResolver::Sequel.new(
|
11
12
|
table_name:,
|
@@ -52,6 +53,10 @@ module TableSync::Receiving::Model
|
|
52
53
|
result
|
53
54
|
end
|
54
55
|
|
56
|
+
def validate_types(data)
|
57
|
+
types_validator.validate(data)
|
58
|
+
end
|
59
|
+
|
55
60
|
def transaction(&)
|
56
61
|
db.transaction(&)
|
57
62
|
end
|
@@ -62,7 +67,7 @@ module TableSync::Receiving::Model
|
|
62
67
|
|
63
68
|
private
|
64
69
|
|
65
|
-
attr_reader :raw_model
|
70
|
+
attr_reader :raw_model, :types_validator
|
66
71
|
|
67
72
|
def dataset
|
68
73
|
raw_model.dataset
|
@@ -0,0 +1,43 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class TableSync::Utils::Schema
|
4
|
+
module Builder
|
5
|
+
class ActiveRecord
|
6
|
+
class << self
|
7
|
+
Type = TableSync::Utils::Schema::Validator::Type
|
8
|
+
|
9
|
+
def build(model)
|
10
|
+
TableSync::Utils::Schema.new(schema(model))
|
11
|
+
end
|
12
|
+
|
13
|
+
private
|
14
|
+
|
15
|
+
def schema(model)
|
16
|
+
model.attribute_types.to_h do |key, value|
|
17
|
+
next [nil, nil] unless (type = type(value))
|
18
|
+
[key.to_sym, TableSync::Utils::Schema::Validator.new(type)]
|
19
|
+
end.tap(&:compact!)
|
20
|
+
end
|
21
|
+
|
22
|
+
def type(value)
|
23
|
+
case value
|
24
|
+
when ActiveModel::Type::String,
|
25
|
+
ActiveModel::Type::ImmutableString
|
26
|
+
Type::STRING
|
27
|
+
when ActiveModel::Type::DateTime,
|
28
|
+
ActiveModel::Type::Date,
|
29
|
+
ActiveModel::Type::Time
|
30
|
+
Type::DATETIME
|
31
|
+
when ActiveModel::Type::Integer,
|
32
|
+
ActiveModel::Type::Decimal,
|
33
|
+
ActiveModel::Type::Float,
|
34
|
+
ActiveModel::Type::BigInteger
|
35
|
+
Type::DECIMAL
|
36
|
+
when ActiveModel::Type::Boolean
|
37
|
+
Type::BOOLEAN
|
38
|
+
end
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
@@ -0,0 +1,37 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class TableSync::Utils::Schema
|
4
|
+
module Builder
|
5
|
+
class Sequel
|
6
|
+
class << self
|
7
|
+
Type = TableSync::Utils::Schema::Validator::Type
|
8
|
+
|
9
|
+
def build(model)
|
10
|
+
TableSync::Utils::Schema.new(schema(model))
|
11
|
+
end
|
12
|
+
|
13
|
+
private
|
14
|
+
|
15
|
+
def schema(model)
|
16
|
+
model.db_schema.transform_values do |value|
|
17
|
+
next unless (type = type(value))
|
18
|
+
TableSync::Utils::Schema::Validator.new(type)
|
19
|
+
end.tap(&:compact!)
|
20
|
+
end
|
21
|
+
|
22
|
+
def type(value)
|
23
|
+
case value[:type]
|
24
|
+
when :string
|
25
|
+
Type::STRING
|
26
|
+
when :datetime, :date, :time
|
27
|
+
Type::DATETIME
|
28
|
+
when :integer, :decimal, :float
|
29
|
+
Type::DECIMAL
|
30
|
+
when :boolean
|
31
|
+
Type::BOOLEAN
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
@@ -0,0 +1,61 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class TableSync::Utils::Schema
|
4
|
+
class Validator
|
5
|
+
class Type
|
6
|
+
class Decimal < Type
|
7
|
+
def valid?(value)
|
8
|
+
!Float(value, exception: false).nil?
|
9
|
+
end
|
10
|
+
end
|
11
|
+
|
12
|
+
class DateTime < Type
|
13
|
+
def valid?(value)
|
14
|
+
Date._parse(value.to_s).any?
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
class Boolean < Type
|
19
|
+
def valid?(value)
|
20
|
+
%w[true false t f 0 1 on off].include?(value.to_s.downcase)
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
class Text < Type
|
25
|
+
def valid?(value)
|
26
|
+
return true if value.is_a?(::String)
|
27
|
+
return true if value.is_a?(::Symbol)
|
28
|
+
return true if Type::DECIMAL.valid?(value)
|
29
|
+
return true if Type::BOOLEAN.valid?(value)
|
30
|
+
return true if Type::DATETIME.valid?(value)
|
31
|
+
false
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
attr_reader :display_name
|
36
|
+
|
37
|
+
def initialize(display_name)
|
38
|
+
@display_name = display_name
|
39
|
+
end
|
40
|
+
|
41
|
+
# @!method valid?
|
42
|
+
|
43
|
+
# rubocop:disable Layout/ClassStructure
|
44
|
+
DECIMAL = Decimal.new("Decimal")
|
45
|
+
DATETIME = DateTime.new("DateTime")
|
46
|
+
BOOLEAN = Boolean.new("Boolean")
|
47
|
+
STRING = Text.new("String")
|
48
|
+
# rubocop:enable Layout/ClassStructure
|
49
|
+
|
50
|
+
def validate(value)
|
51
|
+
return if value.nil?
|
52
|
+
return if valid?(value)
|
53
|
+
"expected #{display_name}, got: #{value.class}"
|
54
|
+
end
|
55
|
+
|
56
|
+
def inspect
|
57
|
+
display_name.inspect
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative "validator/type"
|
4
|
+
|
5
|
+
class TableSync::Utils::Schema
|
6
|
+
class Validator
|
7
|
+
attr_reader :type
|
8
|
+
|
9
|
+
def initialize(type)
|
10
|
+
@type = type
|
11
|
+
end
|
12
|
+
|
13
|
+
def validate(value)
|
14
|
+
type.validate(value)
|
15
|
+
end
|
16
|
+
|
17
|
+
def inspect
|
18
|
+
type.inspect
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Next code checks that hash values have required data types
|
4
|
+
|
5
|
+
require_relative "schema/builder"
|
6
|
+
|
7
|
+
class TableSync::Utils::Schema
|
8
|
+
attr_reader :schema
|
9
|
+
|
10
|
+
def initialize(schema)
|
11
|
+
@schema = schema
|
12
|
+
end
|
13
|
+
|
14
|
+
def validate(data)
|
15
|
+
errors = nil
|
16
|
+
data.each do |row|
|
17
|
+
schema.each do |key, value|
|
18
|
+
if (error = value.validate(row[key]))
|
19
|
+
errors ||= {}
|
20
|
+
errors[key] = error
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
return errors.freeze unless errors.nil?
|
25
|
+
end
|
26
|
+
errors
|
27
|
+
end
|
28
|
+
end
|
data/lib/table_sync/utils.rb
CHANGED
data/lib/table_sync/version.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: table_sync
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 6.
|
4
|
+
version: 6.9.2
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Umbrellio
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2025-
|
11
|
+
date: 2025-10-06 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: memery
|
@@ -128,6 +128,12 @@ files:
|
|
128
128
|
- lib/table_sync/utils/proc_array.rb
|
129
129
|
- lib/table_sync/utils/proc_keywords_resolver.rb
|
130
130
|
- lib/table_sync/utils/required_validator.rb
|
131
|
+
- lib/table_sync/utils/schema.rb
|
132
|
+
- lib/table_sync/utils/schema/builder.rb
|
133
|
+
- lib/table_sync/utils/schema/builder/active_record.rb
|
134
|
+
- lib/table_sync/utils/schema/builder/sequel.rb
|
135
|
+
- lib/table_sync/utils/schema/validator.rb
|
136
|
+
- lib/table_sync/utils/schema/validator/type.rb
|
131
137
|
- lib/table_sync/version.rb
|
132
138
|
- log/.keep
|
133
139
|
- table_sync.gemspec
|
@@ -150,7 +156,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
150
156
|
- !ruby/object:Gem::Version
|
151
157
|
version: '0'
|
152
158
|
requirements: []
|
153
|
-
rubygems_version: 3.
|
159
|
+
rubygems_version: 3.3.27
|
154
160
|
signing_key:
|
155
161
|
specification_version: 4
|
156
162
|
summary: DB Table synchronization between microservices based on Model's event system
|