webflow_sync 0.3.1 → 3.1.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/README.md +75 -4
- data/app/jobs/webflow_sync/create_item_job.rb +10 -2
- data/app/jobs/webflow_sync/initial_sync_job.rb +1 -1
- data/app/jobs/webflow_sync/update_item_job.rb +7 -3
- data/app/services/webflow_sync/api.rb +96 -11
- data/lib/generators/webflow_sync/templates/webflow_sync.rb +2 -0
- data/lib/webflow_sync/configuration.rb +5 -2
- data/lib/webflow_sync/version.rb +1 -1
- metadata +17 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 2c1e92b51a65d85955a3e252f59660dcf59932a7bcca492c7ce439e660efb762
|
4
|
+
data.tar.gz: 3befa85279c19ad561d4b5dd15f1fc2446d89238027969739f98a53966ea6897
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 35bc4a6a9f7b50ff5b1c5062f1554a37f5b2e12e439d8cead8b07188ab70de0d184499ade1caa0b102668d06332515de580507fce0cd9109e1ff17b0ac871710
|
7
|
+
data.tar.gz: fbfa04746a2faa8f34b30879589c2857ecf1ab9bd04b907119f5ad6cce702ec2fc5f7e9d441c3c7145b14cf25607aabd717de45abcb359252ab650dc2f3d0e4b
|
data/README.md
CHANGED
@@ -5,7 +5,9 @@
|
|
5
5
|
|
6
6
|
Keep your Ruby on Rails records in sync with WebFlow.*
|
7
7
|
|
8
|
-
*Currently only one way Rails => WebFlow synchronization
|
8
|
+
*Currently only one way Rails => WebFlow synchronization.
|
9
|
+
|
10
|
+
For the latest changes, check the [CHANGELOG.md](CHANGELOG.md).
|
9
11
|
|
10
12
|
## Installation
|
11
13
|
|
@@ -36,6 +38,35 @@ Run API token Rails generator and follow instructions:
|
|
36
38
|
```bash
|
37
39
|
bundle exec rails generate webflow_sync:api_token_flow
|
38
40
|
```
|
41
|
+
### Configuration options
|
42
|
+
|
43
|
+
In `config/initializers/webflow_sync.rb` you can specify configuration options:
|
44
|
+
|
45
|
+
1. `api_token`
|
46
|
+
2. `webflow_site_id`
|
47
|
+
3. `skip_webflow_sync` - skip synchronization for different environments.
|
48
|
+
4. `publish_on_sync` - republish all domains after create, update and destroy actions.
|
49
|
+
5. `sync_webflow_slug` - save slug generated on WebFlow to the Rails model, `webflow_slug` column.
|
50
|
+
|
51
|
+
This can be useful if you want to link to WebFlow item directly from your Rails app:
|
52
|
+
|
53
|
+
```rb
|
54
|
+
link_to('View on site', "https://#{webflow_domain}/articles/#{record.webflow_slug}", target: :blank)
|
55
|
+
```
|
56
|
+
|
57
|
+
To save slug generated on WebFlow in Rails model, `webflow_slug` column:
|
58
|
+
|
59
|
+
1. add `webflow_slug` column on the model table, then
|
60
|
+
2. set the `sync_webflow_slug` option to `true`.
|
61
|
+
|
62
|
+
Example:
|
63
|
+
|
64
|
+
```rb
|
65
|
+
WebflowSync.configure do |config|
|
66
|
+
config.skip_webflow_sync = ActiveModel::Type::Boolean.new.cast(ENV.fetch('SKIP_WEBFLOW_SYNC'))
|
67
|
+
config.sync_webflow_slug = ActiveModel::Type::Boolean.new.cast(ENV.fetch('SYNC_WEBFLOW_SLUG'))
|
68
|
+
end
|
69
|
+
```
|
39
70
|
|
40
71
|
### Add WebflowSync to models
|
41
72
|
|
@@ -51,17 +82,27 @@ Please note that this _does not_ create a WebFlow collection as that's not possi
|
|
51
82
|
|
52
83
|
As mentioned above, you need to create the WebFlow collection yourself.
|
53
84
|
|
54
|
-
Make sure that the collection `slug` matches the Rails model collection name (the output of `model_class.model_name.
|
85
|
+
Make sure that the collection `slug` matches the Rails model collection name (the output of `model_class.model_name.to_s.underscore.dasherize.pluralize`).
|
55
86
|
|
56
87
|
For example, for `Article` model:
|
57
88
|
|
58
89
|
```ruby
|
59
|
-
> Article.model_name.
|
90
|
+
> Article.model_name.to_s.underscore.dasherize.pluralize
|
60
91
|
# => "articles"
|
61
92
|
```
|
62
93
|
|
63
94
|
Your WebFlow collection `slug` should be `"articles"`.
|
64
95
|
|
96
|
+
For Rails models named with multiple words, make a collection that have a space between words in the name. WebFlow will set the `slug` by replacing space for "-".
|
97
|
+
|
98
|
+
For example, for `FeaturedArticle` model:
|
99
|
+
|
100
|
+
```ruby
|
101
|
+
> FeaturedArticle.model_name.to_s.underscore.dasherize.pluralize
|
102
|
+
# => "featured-articles"
|
103
|
+
```
|
104
|
+
Your WebFlow collection `slug` in this case should be `"featured-articles"`.
|
105
|
+
|
65
106
|
### Set `webflow_site_id`
|
66
107
|
|
67
108
|
There are couple of ways how you can set the `webflow_site_id` to be used.
|
@@ -128,13 +169,43 @@ Or if you already use `#as_json` for some other use-case and cannot modify it, y
|
|
128
169
|
# app/models/article.rb
|
129
170
|
class Article < ApplicationRecord
|
130
171
|
include WebflowSync::ItemSync
|
131
|
-
|
172
|
+
|
132
173
|
def as_webflow_json
|
133
174
|
self.as_json
|
134
175
|
end
|
135
176
|
end
|
136
177
|
```
|
137
178
|
|
179
|
+
### Sync a Rails model with the custom collection
|
180
|
+
|
181
|
+
If collection `slug` does not match the Rails model collection name, you can still sync with WebFlow collection, but need to specify collection slug for each `CreateItemJob` and `UpdateItemJob` call. If not specified, model name is used as a default slug name.
|
182
|
+
You would also need to replace included `WebflowSync::ItemSync` with custom callbacks that will call appropriate WebflowSync jobs.
|
183
|
+
|
184
|
+
For example:
|
185
|
+
|
186
|
+
```ruby
|
187
|
+
WebflowSync::CreateItemJob.perform_later(model_name, id, collection_slug)
|
188
|
+
WebflowSync::UpdateItemJob.perform_later(model_name, id, collection_slug)
|
189
|
+
WebflowSync::DestroyItemJob.perform_later(collection_slug:, webflow_site_id:, webflow_item_id:)
|
190
|
+
```
|
191
|
+
|
192
|
+
Where:
|
193
|
+
|
194
|
+
1. `model_name` - Rails model that has `webflow_site_id` and `webflow_item_id` defined
|
195
|
+
2. `collection_slug` - slug of the WebFlow collection (defaults to: `model_name.underscore.dasherize.pluralize `)
|
196
|
+
|
197
|
+
|
198
|
+
For example:
|
199
|
+
|
200
|
+
```ruby
|
201
|
+
WebflowSync::CreateItemJob.perform_now('articles', 1, 'stories')
|
202
|
+
```
|
203
|
+
Or, if you want to use the default 'articles' collection_slug:
|
204
|
+
|
205
|
+
```ruby
|
206
|
+
WebflowSync::CreateItemJob.perform_now('articles', 1)
|
207
|
+
```
|
208
|
+
|
138
209
|
### Run the initial sync
|
139
210
|
|
140
211
|
After setting up which models you want to sync to WebFlow, you can run the initial sync for each of the models:
|
@@ -2,8 +2,16 @@
|
|
2
2
|
|
3
3
|
module WebflowSync
|
4
4
|
class CreateItemJob < ApplicationJob
|
5
|
-
|
6
|
-
|
5
|
+
# WebFlow collection slug should be in plural form.
|
6
|
+
# For collections that have spaces in their name, WebFlow sets slug by replacing space for "-".
|
7
|
+
# 'JobListing'.underscore.dasherize.pluralize => "job-listings"
|
8
|
+
# 'job_listing'.underscore.dasherize.pluralize => "job-listings"
|
9
|
+
# We can sync Rails model that has different class name than its WebFlow collection
|
10
|
+
# model_name => Rails model that has webflow_site_id and webflow_item_id columns
|
11
|
+
# collection_slug => slug of the WebFlow collection
|
12
|
+
# model_name = 'articles'; id = article.id, collection_slug = 'stories'
|
13
|
+
def perform(model_name, id, collection_slug = model_name.underscore.dasherize.pluralize)
|
14
|
+
model_class = model_name.underscore.classify.constantize
|
7
15
|
record = model_class.find_by(id: id)
|
8
16
|
return if record.blank?
|
9
17
|
return if record.webflow_site_id.blank?
|
@@ -3,7 +3,7 @@
|
|
3
3
|
module WebflowSync
|
4
4
|
class InitialSyncJob < ApplicationJob
|
5
5
|
def perform(collection_slug)
|
6
|
-
model_class = collection_slug.classify.constantize
|
6
|
+
model_class = collection_slug.underscore.classify.constantize
|
7
7
|
model_class.where(webflow_item_id: nil).find_each do |record|
|
8
8
|
next if record.webflow_site_id.blank?
|
9
9
|
|
@@ -2,12 +2,16 @@
|
|
2
2
|
|
3
3
|
module WebflowSync
|
4
4
|
class UpdateItemJob < ApplicationJob
|
5
|
-
|
6
|
-
|
5
|
+
# WebFlow collection slug should be in plural form.
|
6
|
+
# For collections that have spaces in their name, WebFlow sets slug by replacing space for "-".
|
7
|
+
# 'JobListing'.underscore.dasherize.pluralize => 'job-listings'
|
8
|
+
# 'job_listing'.underscore.dasherize.pluralize => 'job-listings'
|
9
|
+
def perform(model_name, id, collection_slug = model_name.underscore.dasherize.pluralize)
|
10
|
+
model_class = model_name.underscore.classify.constantize
|
7
11
|
record = model_class.find_by(id: id)
|
8
12
|
return if record.blank?
|
9
13
|
return if record.webflow_site_id.blank?
|
10
|
-
return WebflowSync::CreateItemJob.perform_now(
|
14
|
+
return WebflowSync::CreateItemJob.perform_now(model_name, id, collection_slug) if record.webflow_item_id.blank?
|
11
15
|
|
12
16
|
WebflowSync::Api.new(record.webflow_site_id).update_item(record, collection_slug)
|
13
17
|
end
|
@@ -8,15 +8,45 @@ module WebflowSync
|
|
8
8
|
@site_id = site_id
|
9
9
|
end
|
10
10
|
|
11
|
+
def get_all_items(collection_slug:, page_limit: 100) # rubocop:disable Metrics/MethodLength
|
12
|
+
collection_id = find_webflow_collection(collection_slug)['_id']
|
13
|
+
max_items_per_page = page_limit # Webflow::Error: 'limit' must be less than or equal to 100
|
14
|
+
first_page_number = 1
|
15
|
+
|
16
|
+
result = make_request(:paginate_items, collection_id, page: first_page_number, per_page: max_items_per_page)
|
17
|
+
puts "Get all items from WebFlow for #{collection_slug} page: #{first_page_number}"
|
18
|
+
|
19
|
+
total_items = result['total']
|
20
|
+
total_pages = (total_items.to_f / max_items_per_page).ceil
|
21
|
+
items = result['items']
|
22
|
+
|
23
|
+
(2..total_pages).each do |page_number|
|
24
|
+
next_page_items = make_request(:paginate_items, collection_id,
|
25
|
+
page: page_number, per_page: max_items_per_page)['items']
|
26
|
+
puts "Get all items from WebFlow for #{collection_slug} page: #{page_number}"
|
27
|
+
|
28
|
+
items.concat next_page_items
|
29
|
+
end
|
30
|
+
|
31
|
+
items
|
32
|
+
end
|
33
|
+
|
34
|
+
def get_item(collection_slug, webflow_item_id)
|
35
|
+
collection = find_webflow_collection(collection_slug)
|
36
|
+
|
37
|
+
make_request(:item, collection['_id'], webflow_item_id)
|
38
|
+
end
|
39
|
+
|
11
40
|
def create_item(record, collection_slug) # rubocop:disable Metrics/MethodLength
|
12
41
|
collection = find_webflow_collection(collection_slug)
|
13
|
-
response =
|
14
|
-
|
15
|
-
|
16
|
-
)
|
42
|
+
response = make_request(:create_item, collection['_id'],
|
43
|
+
record.as_webflow_json.reverse_merge(_archived: false, _draft: false), live: true)
|
44
|
+
|
45
|
+
if update_record_colums(record, response)
|
46
|
+
# When the item is created, sometimes changes are not visible throughout the WebFlow site immediately (probably due to some caching).
|
47
|
+
# To make this change immediately visible from the WebFlow site, we need to publish the site.
|
48
|
+
publish
|
17
49
|
|
18
|
-
# use update_column to skip callbacks to prevent WebflowSync::ItemSync to kick off
|
19
|
-
if record.update_column(:webflow_item_id, response['_id']) # rubocop:disable Rails/SkipsModelValidations
|
20
50
|
puts "Created #{record.inspect} in #{collection_slug}"
|
21
51
|
response
|
22
52
|
else
|
@@ -27,21 +57,37 @@ module WebflowSync
|
|
27
57
|
|
28
58
|
def update_item(record, collection_slug)
|
29
59
|
collection = find_webflow_collection(collection_slug)
|
30
|
-
response =
|
31
|
-
|
32
|
-
|
33
|
-
)
|
60
|
+
response = make_request(:update_item, { '_cid' => collection['_id'], '_id' => record.webflow_item_id },
|
61
|
+
record.as_webflow_json.reverse_merge(_archived: false, _draft: false), live: true)
|
62
|
+
|
63
|
+
# When the item is updated, sometimes changes are not visible throughout the WebFlow site immediately (probably due to some caching).
|
64
|
+
# To make this change immediately visible from the WebFlow site, we need to publish the site.
|
65
|
+
publish
|
66
|
+
|
34
67
|
puts "Updated #{record.inspect} in #{collection_slug}"
|
35
68
|
response
|
36
69
|
end
|
37
70
|
|
38
71
|
def delete_item(collection_slug, webflow_item_id)
|
39
72
|
collection = find_webflow_collection(collection_slug)
|
40
|
-
response =
|
73
|
+
response = make_request(:delete_item, { '_cid' => collection['_id'], '_id' => webflow_item_id })
|
74
|
+
# When the item is removed from WebFlow, it's still visible throughout the WebFlow site (probably due to some caching).
|
75
|
+
# To remove the item immediately from the WebFlow site, we need to publish the site.
|
76
|
+
publish
|
77
|
+
|
41
78
|
puts "Deleted #{webflow_item_id} from #{collection_slug}"
|
42
79
|
response
|
43
80
|
end
|
44
81
|
|
82
|
+
def publish
|
83
|
+
return unless WebflowSync.configuration.publish_on_sync
|
84
|
+
|
85
|
+
response = make_request(:publish, site_id, domain_names: domain_names)
|
86
|
+
|
87
|
+
puts "Publish all domains for Webflow site with id: #{site_id}"
|
88
|
+
response
|
89
|
+
end
|
90
|
+
|
45
91
|
private
|
46
92
|
|
47
93
|
def client
|
@@ -52,11 +98,50 @@ module WebflowSync
|
|
52
98
|
@collections ||= client.collections(site_id)
|
53
99
|
end
|
54
100
|
|
101
|
+
def domain_names
|
102
|
+
@domain_names ||= find_domain_names
|
103
|
+
end
|
104
|
+
|
105
|
+
def find_domain_names
|
106
|
+
# client.domains request does not return the default domain
|
107
|
+
# We need to get default domain name if there are no custom domains set to avoid error:
|
108
|
+
# Webflow::Error: Domain not found for site
|
109
|
+
site = client.site(site_id)
|
110
|
+
default_domain_name = "#{site['shortName']}.webflow.io"
|
111
|
+
names = [default_domain_name]
|
112
|
+
client.domains(site_id).each { |domain| names << domain['name'] }
|
113
|
+
|
114
|
+
names
|
115
|
+
end
|
116
|
+
|
55
117
|
def find_webflow_collection(collection_slug)
|
56
118
|
response = collections.find { |collection| collection['slug'] == collection_slug }
|
57
119
|
raise "Cannot find collection #{collection_slug} for Webflow site #{site_id}" unless response
|
58
120
|
|
59
121
|
response
|
60
122
|
end
|
123
|
+
|
124
|
+
def update_record_colums(record, response)
|
125
|
+
# use update_column to skip callbacks to prevent WebflowSync::ItemSync to kick off
|
126
|
+
if WebflowSync.configuration.sync_webflow_slug
|
127
|
+
record.update_columns(webflow_item_id: response['_id'], webflow_slug: response['slug']) # rubocop:disable Rails/SkipsModelValidations
|
128
|
+
else
|
129
|
+
record.update_column(:webflow_item_id, response['_id']) # rubocop:disable Rails/SkipsModelValidations
|
130
|
+
end
|
131
|
+
end
|
132
|
+
|
133
|
+
def make_request(method_name, *args, retries: 0, **kwargs)
|
134
|
+
if kwargs.present?
|
135
|
+
client.public_send(method_name, *args, **kwargs)
|
136
|
+
else
|
137
|
+
client.public_send(method_name, *args)
|
138
|
+
end
|
139
|
+
rescue Webflow::Error => e
|
140
|
+
raise if retries >= 8 || e.message.strip != 'Rate limit hit'
|
141
|
+
|
142
|
+
puts "Sleeping #{2**retries} seconds"
|
143
|
+
sleep 2**retries
|
144
|
+
make_request(method_name, *args, retries: retries + 1, **kwargs)
|
145
|
+
end
|
61
146
|
end
|
62
147
|
end
|
@@ -4,4 +4,6 @@ WebflowSync.configure do |config|
|
|
4
4
|
# config.api_token = ENV.fetch('WEBFLOW_API_TOKEN')
|
5
5
|
# config.skip_webflow_sync = Rails.env.test? # default
|
6
6
|
config.webflow_site_id = ENV.fetch('WEBFLOW_SITE_ID')
|
7
|
+
config.sync_webflow_slug = ENV.fetch('SYNC_WEBFLOW_SLUG')
|
8
|
+
config.publish_on_sync = true
|
7
9
|
end
|
@@ -6,10 +6,13 @@ module WebflowSync
|
|
6
6
|
|
7
7
|
def configure
|
8
8
|
self.configuration ||= Configuration.new
|
9
|
+
|
10
|
+
self.configuration.publish_on_sync = true
|
11
|
+
self.configuration.skip_webflow_sync = !Rails.env.production?
|
12
|
+
|
9
13
|
yield(self.configuration)
|
10
14
|
|
11
15
|
self.configuration.api_token ||= ENV.fetch('WEBFLOW_API_TOKEN')
|
12
|
-
defined?(self.configuration.skip_webflow_sync) or self.configuration.skip_webflow_sync = !Rails.env.production?
|
13
16
|
end
|
14
17
|
|
15
18
|
private
|
@@ -18,7 +21,7 @@ module WebflowSync
|
|
18
21
|
end
|
19
22
|
|
20
23
|
class Configuration
|
21
|
-
attr_accessor :skip_webflow_sync, :webflow_site_id
|
24
|
+
attr_accessor :skip_webflow_sync, :webflow_site_id, :sync_webflow_slug, :publish_on_sync
|
22
25
|
attr_reader :api_token
|
23
26
|
|
24
27
|
def api_token=(value)
|
data/lib/webflow_sync/version.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: webflow_sync
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version:
|
4
|
+
version: 3.1.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Viktor
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2021-
|
11
|
+
date: 2021-07-06 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rails
|
@@ -38,6 +38,20 @@ dependencies:
|
|
38
38
|
- - ">="
|
39
39
|
- !ruby/object:Gem::Version
|
40
40
|
version: '0'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: dotenv-rails
|
43
|
+
requirement: !ruby/object:Gem::Requirement
|
44
|
+
requirements:
|
45
|
+
- - ">="
|
46
|
+
- !ruby/object:Gem::Version
|
47
|
+
version: '0'
|
48
|
+
type: :development
|
49
|
+
prerelease: false
|
50
|
+
version_requirements: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - ">="
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: '0'
|
41
55
|
- !ruby/object:Gem::Dependency
|
42
56
|
name: factory_bot_rails
|
43
57
|
requirement: !ruby/object:Gem::Requirement
|
@@ -173,7 +187,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
173
187
|
- !ruby/object:Gem::Version
|
174
188
|
version: '0'
|
175
189
|
requirements: []
|
176
|
-
rubygems_version: 3.
|
190
|
+
rubygems_version: 3.0.9
|
177
191
|
signing_key:
|
178
192
|
specification_version: 4
|
179
193
|
summary: Keep Rails models in sync with WebFlow.
|