graphql-anycable 0.3.0 → 0.4.1
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/.github/ISSUE_TEMPLATE/bug_report.md +63 -0
- data/.github/ISSUE_TEMPLATE/feature_request.md +20 -0
- data/.gitignore +1 -2
- data/.travis.yml +30 -5
- data/Appraisals +19 -0
- data/CHANGELOG.md +59 -0
- data/Gemfile +3 -0
- data/README.md +35 -2
- data/gemfiles/.bundle/config +2 -0
- data/gemfiles/anycable_0.6.gemfile +14 -0
- data/gemfiles/anycable_1.0.gemfile +14 -0
- data/gemfiles/graphql_1.10.gemfile +14 -0
- data/gemfiles/graphql_1.11.gemfile +14 -0
- data/gemfiles/graphql_1.9.gemfile +14 -0
- data/graphql-anycable.gemspec +6 -4
- data/lib/graphql-anycable.rb +11 -2
- data/lib/graphql/anycable/cleaner.rb +66 -0
- data/lib/graphql/anycable/config.rb +1 -0
- data/lib/graphql/anycable/tasks/clean_expired_subscriptions.rake +16 -28
- data/lib/graphql/anycable/version.rb +1 -1
- data/lib/graphql/subscriptions/anycable_subscriptions.rb +9 -12
- metadata +64 -14
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: f463febbc62debd78ca0f3beffb7a6e97f9f10a850256cd97ad49f3af4894587
|
4
|
+
data.tar.gz: 8d10b35e3bed8d8e8f0524540cc1a049d6af9ed7673f5335dec09e9a0deb420d
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 9f0f7636f5062096548548acfb4dd16d7762d55b39d0e53fb750f5ae549936adb9511cc179cf06becdf2e67b513f9d3ebb15b44d3e9dd9f6146da99087ce801c
|
7
|
+
data.tar.gz: 94880103ba2deb26aa42549591e18aeee903c7fc33a09a5f26c3498174250ae583acccc1ded333890a6150846cc91dd975a5ee187c16eaebca3de82f996a23a2
|
@@ -0,0 +1,63 @@
|
|
1
|
+
---
|
2
|
+
name: Bug report
|
3
|
+
about: Create a report to help us improve graphql-anycable
|
4
|
+
title: ''
|
5
|
+
labels: ''
|
6
|
+
assignees: ''
|
7
|
+
|
8
|
+
---
|
9
|
+
|
10
|
+
**Describe the bug**
|
11
|
+
A clear and concise description of what the bug is.
|
12
|
+
|
13
|
+
**Versions**
|
14
|
+
ruby:
|
15
|
+
rails (or other framework):
|
16
|
+
graphql:
|
17
|
+
graphql-anycable:
|
18
|
+
anycable:
|
19
|
+
|
20
|
+
**GraphQL schema**
|
21
|
+
Provide relevant details. Are you using [subscription classes](https://graphql-ruby.org/subscriptions/subscription_classes.html) or not (graphql-ruby behavior differs there)?
|
22
|
+
|
23
|
+
```ruby
|
24
|
+
class Product < GraphQL::Schema::Object
|
25
|
+
field :id, ID, null: false, hash_key: :id
|
26
|
+
field :title, String, null: true, hash_key: :title
|
27
|
+
end
|
28
|
+
|
29
|
+
class SubscriptionType < GraphQL::Schema::Object
|
30
|
+
field :product_created, Product, null: false
|
31
|
+
field :product_updated, Product, null: false
|
32
|
+
|
33
|
+
def product_created; end
|
34
|
+
def product_updated; end
|
35
|
+
end
|
36
|
+
|
37
|
+
class ApplicationSchema < GraphQL::Schema
|
38
|
+
subscription SubscriptionType
|
39
|
+
end
|
40
|
+
```
|
41
|
+
|
42
|
+
**GraphQL query**
|
43
|
+
|
44
|
+
How do you subscribe to subscriptions?
|
45
|
+
|
46
|
+
```graphql
|
47
|
+
subscription {
|
48
|
+
productCreated { id title }
|
49
|
+
productUpdated { id }
|
50
|
+
}
|
51
|
+
```
|
52
|
+
|
53
|
+
**Steps to reproduce**
|
54
|
+
Steps to reproduce the behavior
|
55
|
+
|
56
|
+
**Expected behavior**
|
57
|
+
A clear and concise description of what you expected to happen.
|
58
|
+
|
59
|
+
**Actual behavior**
|
60
|
+
What specifically went wrong?
|
61
|
+
|
62
|
+
**Additional context**
|
63
|
+
Add any other context about the problem here. Tracebacks, your thoughts. anything that may be useful.
|
@@ -0,0 +1,20 @@
|
|
1
|
+
---
|
2
|
+
name: Feature request
|
3
|
+
about: Suggest an idea for this project
|
4
|
+
title: ''
|
5
|
+
labels: ''
|
6
|
+
assignees: ''
|
7
|
+
|
8
|
+
---
|
9
|
+
|
10
|
+
**Is your feature request related to a problem? Please describe.**
|
11
|
+
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
12
|
+
|
13
|
+
**Describe the solution you'd like**
|
14
|
+
A clear and concise description of what you want to happen.
|
15
|
+
|
16
|
+
**Describe alternatives you've considered**
|
17
|
+
A clear and concise description of any alternative solutions or features you've considered.
|
18
|
+
|
19
|
+
**Additional context**
|
20
|
+
Add any other context or screenshots about the feature request here.
|
data/.gitignore
CHANGED
data/.travis.yml
CHANGED
@@ -1,9 +1,34 @@
|
|
1
1
|
---
|
2
|
-
sudo: false
|
3
2
|
language: ruby
|
4
3
|
cache: bundler
|
5
4
|
rvm:
|
6
|
-
- 2.
|
7
|
-
- 2.
|
8
|
-
- 2.
|
9
|
-
|
5
|
+
- 2.7.1
|
6
|
+
- 2.6.6
|
7
|
+
- 2.5.8
|
8
|
+
gemfile:
|
9
|
+
- gemfiles/graphql_1.9.gemfile
|
10
|
+
- gemfiles/graphql_1.10.gemfile
|
11
|
+
- gemfiles/graphql_1.11.gemfile
|
12
|
+
- gemfiles/anycable_0.6.gemfile
|
13
|
+
- gemfiles/anycable_1.0.gemfile
|
14
|
+
env:
|
15
|
+
- GRAPHQL_RUBY_INTERPRETER=yes
|
16
|
+
- GRAPHQL_RUBY_INTERPRETER=no
|
17
|
+
before_install: gem install bundler -v "~> 2.0"
|
18
|
+
|
19
|
+
jobs:
|
20
|
+
exclude:
|
21
|
+
# Exclude new dependencies on old rubies to run less jobs
|
22
|
+
- rvm: 2.5.8
|
23
|
+
gemfile: gemfiles/anycable_1.0.gemfile
|
24
|
+
- rvm: 2.5.8
|
25
|
+
gemfile: gemfiles/graphql_1.10.gemfile
|
26
|
+
- rvm: 2.5.8
|
27
|
+
gemfile: gemfiles/graphql_1.11.gemfile
|
28
|
+
# Exclude old dependencies on new rubies to run less jobs
|
29
|
+
- rvm: 2.6.5
|
30
|
+
gemfile: gemfiles/anycable_0.6.gemfile
|
31
|
+
- rvm: 2.6.5
|
32
|
+
gemfile: gemfiles/graphql_1.9.gemfile
|
33
|
+
- rvm: 2.6.5
|
34
|
+
gemfile: gemfiles/graphql_1.10.gemfile
|
data/Appraisals
ADDED
@@ -0,0 +1,19 @@
|
|
1
|
+
appraise "graphql-1.9" do
|
2
|
+
gem "graphql", "~> 1.9.0"
|
3
|
+
end
|
4
|
+
|
5
|
+
appraise "graphql-1.10" do
|
6
|
+
gem "graphql", "~> 1.10.0"
|
7
|
+
end
|
8
|
+
|
9
|
+
appraise "graphql-1.11" do
|
10
|
+
gem "graphql", "~> 1.11.0"
|
11
|
+
end
|
12
|
+
|
13
|
+
appraise "anycable-0.6" do
|
14
|
+
gem "anycable", "~> 0.6.0"
|
15
|
+
end
|
16
|
+
|
17
|
+
appraise "anycable-1.0" do
|
18
|
+
gem "anycable", "~> 1.0.0.preview1"
|
19
|
+
end
|
data/CHANGELOG.md
ADDED
@@ -0,0 +1,59 @@
|
|
1
|
+
# Changelog
|
2
|
+
|
3
|
+
All notable changes to this project will be documented in this file.
|
4
|
+
|
5
|
+
The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/)
|
6
|
+
and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html).
|
7
|
+
|
8
|
+
## 0.4.1 - 2020-08-21
|
9
|
+
|
10
|
+
### Fixed
|
11
|
+
|
12
|
+
- Deprecation warning for `Redis#exist` usage on Redis Ruby client 4.2+. Switch to `exists?` method and require Redis 4.2+ (see [#14](https://github.com/anycable/graphql-anycable/issues/14)). [@Envek]
|
13
|
+
|
14
|
+
## 0.4.0 - 2020-03-19
|
15
|
+
|
16
|
+
### Added
|
17
|
+
|
18
|
+
- Ability to configure the gem via `configure` block, in addition to enironment variables and yaml files. [@gsamokovarov] ([#11](https://github.com/Envek/graphql-anycable/pull/11))
|
19
|
+
- Ability to run Redis cleaning operations outside of Rake. [@gsamokovarov] ([#11](https://github.com/Envek/graphql-anycable/pull/11))
|
20
|
+
- AnyCable 1.0 compatibility. [@bibendi], [@Envek] ([#10](https://github.com/Envek/graphql-anycable/pull/10))
|
21
|
+
|
22
|
+
## 0.3.3 - 2020-03-03
|
23
|
+
|
24
|
+
### Fixed
|
25
|
+
|
26
|
+
- Redis command error on subscription query with multiple fields. [@Envek] ([#9](https://github.com/Envek/graphql-anycable/issues/9))
|
27
|
+
|
28
|
+
## 0.3.2 - 2020-03-02
|
29
|
+
|
30
|
+
### Added
|
31
|
+
|
32
|
+
- Ability to skip some cleanup on restricted Redis instances (like Heroku). [@Envek] ([#8](https://github.com/Envek/graphql-anycable/issues/8))
|
33
|
+
|
34
|
+
## 0.3.1 - 2019-06-13
|
35
|
+
|
36
|
+
### Fixed
|
37
|
+
|
38
|
+
- Empty operation name handling. [@FX-HAO] ([#3](https://github.com/Envek/graphql-anycable/pull/3))
|
39
|
+
|
40
|
+
## 0.3.0 - 2018-11-16
|
41
|
+
|
42
|
+
### Added
|
43
|
+
|
44
|
+
- AnyCable 0.6 compatibility. [@Envek]
|
45
|
+
|
46
|
+
## 0.2.0 - 2018-09-17
|
47
|
+
|
48
|
+
### Added
|
49
|
+
|
50
|
+
- Subscription expiration and rake task for stale subscriptions cleanup. [@Envek]
|
51
|
+
|
52
|
+
### 0.1.0 - 2018-08-26
|
53
|
+
|
54
|
+
Initial version: store subscriptions on redis, re-execute queries in sync. [@Envek]
|
55
|
+
|
56
|
+
[@gsamokovarov]: https://github.com/gsamokovarov "Genadi Samokovarov"
|
57
|
+
[@bibendi]: https://github.com/bibendi "Misha Merkushin"
|
58
|
+
[@FX-HAO]: https://github.com/FX-HAO "Fuxin Hao"
|
59
|
+
[@Envek]: https://github.com/Envek "Andrey Novikov"
|
data/Gemfile
CHANGED
data/README.md
CHANGED
@@ -103,8 +103,41 @@ Or install it yourself as:
|
|
103
103
|
|
104
104
|
To avoid filling Redis storage with stale subscription data:
|
105
105
|
|
106
|
-
1. Set `
|
107
|
-
|
106
|
+
1. Set `subscription_expiration_seconds` setting to number of seconds (e.g. `604800` for 1 week). See [configuration](#Configuration) section below for details.
|
107
|
+
|
108
|
+
2. Execute `rake graphql:anycable:clean` once in a while to clean up stale subscription data.
|
109
|
+
|
110
|
+
Heroku users should set up `use_redis_object_on_cleanup` setting to `false` due to [limitations in Heroku Redis](https://devcenter.heroku.com/articles/heroku-redis#connection-permissions).
|
111
|
+
|
112
|
+
## Configuration
|
113
|
+
|
114
|
+
GraphQL-Anycable uses [anyway_config] to configure itself. There are several possibilities to configure this gem:
|
115
|
+
|
116
|
+
1. Environment variables:
|
117
|
+
|
118
|
+
```.env
|
119
|
+
GRAPHQL_ANYCABLE_SUBSCRIPTION_EXPIRATION_SECONDS=604800
|
120
|
+
GRAPHQL_ANYCABLE_USE_REDIS_OBJECT_ON_CLEANUP=true
|
121
|
+
```
|
122
|
+
|
123
|
+
2. YAML configuration files:
|
124
|
+
|
125
|
+
```yaml
|
126
|
+
# config/graphql_anycable.yml
|
127
|
+
production:
|
128
|
+
subscription_expiration_seconds: 300 # 5 minutes
|
129
|
+
use_redis_object_on_cleanup: false # For restricted redis installations
|
130
|
+
```
|
131
|
+
|
132
|
+
3. Configuration from your application code:
|
133
|
+
|
134
|
+
```ruby
|
135
|
+
GraphQL::Anycable.configure do |config|
|
136
|
+
config.subscription_expiration_seconds = 3600 # 1 hour
|
137
|
+
end
|
138
|
+
```
|
139
|
+
|
140
|
+
And any other way provided by [anyway_config]. Check its documentation!
|
108
141
|
|
109
142
|
## Development
|
110
143
|
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# This file was generated by Appraisal
|
2
|
+
|
3
|
+
source "https://rubygems.org"
|
4
|
+
|
5
|
+
gem "anycable", "~> 0.6.0"
|
6
|
+
|
7
|
+
group :development, :test do
|
8
|
+
gem "pry"
|
9
|
+
gem "pry-byebug", platform: :mri
|
10
|
+
gem "rubocop"
|
11
|
+
gem "rubocop-rspec"
|
12
|
+
end
|
13
|
+
|
14
|
+
gemspec path: "../"
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# This file was generated by Appraisal
|
2
|
+
|
3
|
+
source "https://rubygems.org"
|
4
|
+
|
5
|
+
gem "anycable", "~> 1.0.0.preview1"
|
6
|
+
|
7
|
+
group :development, :test do
|
8
|
+
gem "pry"
|
9
|
+
gem "pry-byebug", platform: :mri
|
10
|
+
gem "rubocop"
|
11
|
+
gem "rubocop-rspec"
|
12
|
+
end
|
13
|
+
|
14
|
+
gemspec path: "../"
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# This file was generated by Appraisal
|
2
|
+
|
3
|
+
source "https://rubygems.org"
|
4
|
+
|
5
|
+
gem "graphql", "~> 1.10.0"
|
6
|
+
|
7
|
+
group :development, :test do
|
8
|
+
gem "pry"
|
9
|
+
gem "pry-byebug", platform: :mri
|
10
|
+
gem "rubocop"
|
11
|
+
gem "rubocop-rspec"
|
12
|
+
end
|
13
|
+
|
14
|
+
gemspec path: "../"
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# This file was generated by Appraisal
|
2
|
+
|
3
|
+
source "https://rubygems.org"
|
4
|
+
|
5
|
+
gem "graphql", "~> 1.11.0"
|
6
|
+
|
7
|
+
group :development, :test do
|
8
|
+
gem "pry"
|
9
|
+
gem "pry-byebug", platform: :mri
|
10
|
+
gem "rubocop"
|
11
|
+
gem "rubocop-rspec"
|
12
|
+
end
|
13
|
+
|
14
|
+
gemspec path: "../"
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# This file was generated by Appraisal
|
2
|
+
|
3
|
+
source "https://rubygems.org"
|
4
|
+
|
5
|
+
gem "graphql", "~> 1.9.0"
|
6
|
+
|
7
|
+
group :development, :test do
|
8
|
+
gem "pry"
|
9
|
+
gem "pry-byebug", platform: :mri
|
10
|
+
gem "rubocop"
|
11
|
+
gem "rubocop-rspec"
|
12
|
+
end
|
13
|
+
|
14
|
+
gemspec path: "../"
|
data/graphql-anycable.gemspec
CHANGED
@@ -26,12 +26,14 @@ Gem::Specification.new do |spec|
|
|
26
26
|
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
27
27
|
spec.require_paths = ["lib"]
|
28
28
|
|
29
|
-
spec.add_dependency "anycable", "
|
30
|
-
spec.add_dependency "anyway_config", "
|
29
|
+
spec.add_dependency "anycable", ">= 0.6.0", "< 2"
|
30
|
+
spec.add_dependency "anyway_config", ">= 1.3", "< 3"
|
31
31
|
spec.add_dependency "graphql", "~> 1.8"
|
32
|
+
spec.add_dependency "redis", ">= 4.2.0"
|
32
33
|
|
33
|
-
spec.add_development_dependency "bundler", "~>
|
34
|
+
spec.add_development_dependency "bundler", "~> 2.0"
|
34
35
|
spec.add_development_dependency "fakeredis"
|
35
|
-
spec.add_development_dependency "rake", "
|
36
|
+
spec.add_development_dependency "rake", ">= 12.3.3"
|
36
37
|
spec.add_development_dependency "rspec", "~> 3.0"
|
38
|
+
spec.add_development_dependency "appraisal"
|
37
39
|
end
|
data/lib/graphql-anycable.rb
CHANGED
@@ -3,6 +3,7 @@
|
|
3
3
|
require "graphql"
|
4
4
|
|
5
5
|
require_relative "graphql/anycable/version"
|
6
|
+
require_relative "graphql/anycable/cleaner"
|
6
7
|
require_relative "graphql/anycable/config"
|
7
8
|
require_relative "graphql/anycable/railtie" if defined?(Rails)
|
8
9
|
require_relative "graphql/subscriptions/anycable_subscriptions"
|
@@ -13,13 +14,21 @@ module GraphQL
|
|
13
14
|
|
14
15
|
def redis
|
15
16
|
@redis ||= begin
|
16
|
-
adapter = ::
|
17
|
+
adapter = ::AnyCable.broadcast_adapter
|
17
18
|
unless adapter.is_a?(::AnyCable::BroadcastAdapters::Redis)
|
18
19
|
raise "Unsupported AnyCable adapter: #{adapter.class}. " \
|
19
20
|
"graphql-anycable works only with redis broadcast adapter."
|
20
21
|
end
|
21
|
-
::
|
22
|
+
::AnyCable.broadcast_adapter.redis_conn
|
22
23
|
end
|
23
24
|
end
|
25
|
+
|
26
|
+
def config
|
27
|
+
@config ||= GraphQL::Anycable::Config.new
|
28
|
+
end
|
29
|
+
|
30
|
+
def configure
|
31
|
+
yield(config) if block_given?
|
32
|
+
end
|
24
33
|
end
|
25
34
|
end
|
@@ -0,0 +1,66 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module GraphQL
|
4
|
+
module Anycable
|
5
|
+
module Cleaner
|
6
|
+
extend self
|
7
|
+
|
8
|
+
def clean
|
9
|
+
clean_channels
|
10
|
+
clean_subscriptions
|
11
|
+
clean_events
|
12
|
+
end
|
13
|
+
|
14
|
+
def clean_channels
|
15
|
+
return unless config.subscription_expiration_seconds
|
16
|
+
return unless config.use_redis_object_on_cleanup
|
17
|
+
|
18
|
+
redis.scan_each(match: "#{adapter::CHANNEL_PREFIX}*") do |key|
|
19
|
+
idle = redis.object("IDLETIME", key)
|
20
|
+
next if idle&.<= config.subscription_expiration_seconds
|
21
|
+
|
22
|
+
redis.del(key)
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
def clean_subscriptions
|
27
|
+
return unless config.subscription_expiration_seconds
|
28
|
+
return unless config.use_redis_object_on_cleanup
|
29
|
+
|
30
|
+
redis.scan_each(match: "#{adapter::SUBSCRIPTION_PREFIX}*") do |key|
|
31
|
+
idle = redis.object("IDLETIME", key)
|
32
|
+
next if idle&.<= config.subscription_expiration_seconds
|
33
|
+
|
34
|
+
redis.del(key)
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
def clean_events
|
39
|
+
redis.scan_each(match: "#{adapter::SUBSCRIPTION_EVENTS_PREFIX}*") do |key|
|
40
|
+
subscription_id = key.sub(/\A#{adapter::SUBSCRIPTION_EVENTS_PREFIX}/, "")
|
41
|
+
next if redis.exists?(adapter::SUBSCRIPTION_PREFIX + subscription_id)
|
42
|
+
|
43
|
+
redis.smembers(key).each do |event_topic|
|
44
|
+
redis.srem(adapter::EVENT_PREFIX + event_topic, subscription_id)
|
45
|
+
end
|
46
|
+
|
47
|
+
redis.del(key)
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
private
|
52
|
+
|
53
|
+
def adapter
|
54
|
+
GraphQL::Subscriptions::AnyCableSubscriptions
|
55
|
+
end
|
56
|
+
|
57
|
+
def redis
|
58
|
+
GraphQL::Anycable.redis
|
59
|
+
end
|
60
|
+
|
61
|
+
def config
|
62
|
+
GraphQL::Anycable.config
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
66
|
+
end
|
@@ -1,41 +1,29 @@
|
|
1
|
-
require "graphql-anycable"
|
2
|
-
|
3
1
|
# frozen_string_literal: true
|
4
2
|
|
3
|
+
require "graphql-anycable"
|
4
|
+
|
5
5
|
namespace :graphql do
|
6
6
|
namespace :anycable do
|
7
|
-
|
8
|
-
|
9
|
-
unless config.subscription_expiration_seconds
|
10
|
-
warn "GraphQL::Anycable: No expiration set for subscriptions!"
|
11
|
-
next
|
12
|
-
end
|
7
|
+
desc "Clean up stale graphql channels, subscriptions, and events from redis"
|
8
|
+
task clean: %i[clean:channels clean:subscriptions clean:events]
|
13
9
|
|
14
|
-
|
15
|
-
|
10
|
+
# Old name that was used earlier
|
11
|
+
task clean_expired_subscriptions: :clean
|
16
12
|
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
redis.del(key)
|
13
|
+
namespace :clean do
|
14
|
+
# Clean up old channels
|
15
|
+
task :channels do
|
16
|
+
GraphQL::Anycable::Cleaner.clean_channels
|
22
17
|
end
|
23
18
|
|
24
|
-
#
|
25
|
-
|
26
|
-
|
27
|
-
next if idle&.<= config.subscription_expiration_seconds
|
28
|
-
redis.del(key)
|
19
|
+
# Clean up old subscriptions (they should have expired by themselves)
|
20
|
+
task :subscriptions do
|
21
|
+
GraphQL::Anycable::Cleaner.clean_subscriptions
|
29
22
|
end
|
30
23
|
|
31
|
-
#
|
32
|
-
|
33
|
-
|
34
|
-
next if redis.exists(klass::SUBSCRIPTION_PREFIX + subscription_id)
|
35
|
-
redis.smembers(key).each do |event_topic|
|
36
|
-
redis.srem(klass::EVENT_PREFIX + event_topic, subscription_id)
|
37
|
-
end
|
38
|
-
redis.del(key)
|
24
|
+
# Clean up subscription_ids from events for expired subscriptions
|
25
|
+
task :events do
|
26
|
+
GraphQL::Anycable::Cleaner.clean_events
|
39
27
|
end
|
40
28
|
end
|
41
29
|
end
|
@@ -54,7 +54,7 @@ module GraphQL
|
|
54
54
|
class AnyCableSubscriptions < GraphQL::Subscriptions
|
55
55
|
extend Forwardable
|
56
56
|
|
57
|
-
def_delegators :"GraphQL::Anycable", :redis
|
57
|
+
def_delegators :"GraphQL::Anycable", :redis, :config
|
58
58
|
|
59
59
|
SUBSCRIPTION_PREFIX = "graphql-subscription:"
|
60
60
|
SUBSCRIPTION_EVENTS_PREFIX = "graphql-subscription-events:"
|
@@ -71,7 +71,7 @@ module GraphQL
|
|
71
71
|
# Re-evaluate all subscribed queries and push the data over ActionCable.
|
72
72
|
def execute_all(event, object)
|
73
73
|
redis.smembers(EVENT_PREFIX + event.topic).each do |subscription_id|
|
74
|
-
next unless redis.exists(SUBSCRIPTION_PREFIX + subscription_id)
|
74
|
+
next unless redis.exists?(SUBSCRIPTION_PREFIX + subscription_id)
|
75
75
|
execute(subscription_id, event, object)
|
76
76
|
end
|
77
77
|
end
|
@@ -79,7 +79,7 @@ module GraphQL
|
|
79
79
|
# This subscription was re-evaluated.
|
80
80
|
# Send it to the specific stream where this client was waiting.
|
81
81
|
def deliver(subscription_id, result)
|
82
|
-
payload = {
|
82
|
+
payload = {result: result.to_h, more: true}
|
83
83
|
anycable.broadcast(SUBSCRIPTION_PREFIX + subscription_id, payload.to_json)
|
84
84
|
end
|
85
85
|
|
@@ -95,13 +95,13 @@ module GraphQL
|
|
95
95
|
query_string: query.query_string,
|
96
96
|
variables: query.provided_variables.to_json,
|
97
97
|
context: @serializer.dump(context.to_h),
|
98
|
-
operation_name: query.operation_name
|
98
|
+
operation_name: query.operation_name
|
99
99
|
}
|
100
100
|
|
101
101
|
redis.multi do
|
102
102
|
redis.sadd(CHANNEL_PREFIX + channel.params["channelId"], subscription_id)
|
103
103
|
redis.mapped_hmset(SUBSCRIPTION_PREFIX + subscription_id, data)
|
104
|
-
redis.sadd(SUBSCRIPTION_EVENTS_PREFIX + subscription_id,
|
104
|
+
redis.sadd(SUBSCRIPTION_EVENTS_PREFIX + subscription_id, events.map(&:topic))
|
105
105
|
events.each do |event|
|
106
106
|
redis.sadd(EVENT_PREFIX + event.topic, subscription_id)
|
107
107
|
end
|
@@ -115,10 +115,11 @@ module GraphQL
|
|
115
115
|
def read_subscription(subscription_id)
|
116
116
|
redis.mapped_hmget(
|
117
117
|
"#{SUBSCRIPTION_PREFIX}#{subscription_id}",
|
118
|
-
:query_string, :variables, :context, :operation_name
|
118
|
+
:query_string, :variables, :context, :operation_name
|
119
119
|
).tap do |subscription|
|
120
|
-
subscription[:context]
|
120
|
+
subscription[:context] = @serializer.load(subscription[:context])
|
121
121
|
subscription[:variables] = JSON.parse(subscription[:variables])
|
122
|
+
subscription[:operation_name] = nil if subscription[:operation_name].strip == ""
|
122
123
|
end
|
123
124
|
end
|
124
125
|
|
@@ -144,11 +145,7 @@ module GraphQL
|
|
144
145
|
private
|
145
146
|
|
146
147
|
def anycable
|
147
|
-
@anycable ||= ::
|
148
|
-
end
|
149
|
-
|
150
|
-
def config
|
151
|
-
@config ||= GraphQL::Anycable::Config.new
|
148
|
+
@anycable ||= ::AnyCable.broadcast_adapter
|
152
149
|
end
|
153
150
|
end
|
154
151
|
end
|
metadata
CHANGED
@@ -1,43 +1,55 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: graphql-anycable
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.4.1
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Andrey Novikov
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2020-08-21 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: anycable
|
15
15
|
requirement: !ruby/object:Gem::Requirement
|
16
16
|
requirements:
|
17
|
-
- - "
|
17
|
+
- - ">="
|
18
18
|
- !ruby/object:Gem::Version
|
19
19
|
version: 0.6.0
|
20
|
+
- - "<"
|
21
|
+
- !ruby/object:Gem::Version
|
22
|
+
version: '2'
|
20
23
|
type: :runtime
|
21
24
|
prerelease: false
|
22
25
|
version_requirements: !ruby/object:Gem::Requirement
|
23
26
|
requirements:
|
24
|
-
- - "
|
27
|
+
- - ">="
|
25
28
|
- !ruby/object:Gem::Version
|
26
29
|
version: 0.6.0
|
30
|
+
- - "<"
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: '2'
|
27
33
|
- !ruby/object:Gem::Dependency
|
28
34
|
name: anyway_config
|
29
35
|
requirement: !ruby/object:Gem::Requirement
|
30
36
|
requirements:
|
31
|
-
- - "
|
37
|
+
- - ">="
|
32
38
|
- !ruby/object:Gem::Version
|
33
39
|
version: '1.3'
|
40
|
+
- - "<"
|
41
|
+
- !ruby/object:Gem::Version
|
42
|
+
version: '3'
|
34
43
|
type: :runtime
|
35
44
|
prerelease: false
|
36
45
|
version_requirements: !ruby/object:Gem::Requirement
|
37
46
|
requirements:
|
38
|
-
- - "
|
47
|
+
- - ">="
|
39
48
|
- !ruby/object:Gem::Version
|
40
49
|
version: '1.3'
|
50
|
+
- - "<"
|
51
|
+
- !ruby/object:Gem::Version
|
52
|
+
version: '3'
|
41
53
|
- !ruby/object:Gem::Dependency
|
42
54
|
name: graphql
|
43
55
|
requirement: !ruby/object:Gem::Requirement
|
@@ -52,20 +64,34 @@ dependencies:
|
|
52
64
|
- - "~>"
|
53
65
|
- !ruby/object:Gem::Version
|
54
66
|
version: '1.8'
|
67
|
+
- !ruby/object:Gem::Dependency
|
68
|
+
name: redis
|
69
|
+
requirement: !ruby/object:Gem::Requirement
|
70
|
+
requirements:
|
71
|
+
- - ">="
|
72
|
+
- !ruby/object:Gem::Version
|
73
|
+
version: 4.2.0
|
74
|
+
type: :runtime
|
75
|
+
prerelease: false
|
76
|
+
version_requirements: !ruby/object:Gem::Requirement
|
77
|
+
requirements:
|
78
|
+
- - ">="
|
79
|
+
- !ruby/object:Gem::Version
|
80
|
+
version: 4.2.0
|
55
81
|
- !ruby/object:Gem::Dependency
|
56
82
|
name: bundler
|
57
83
|
requirement: !ruby/object:Gem::Requirement
|
58
84
|
requirements:
|
59
85
|
- - "~>"
|
60
86
|
- !ruby/object:Gem::Version
|
61
|
-
version: '
|
87
|
+
version: '2.0'
|
62
88
|
type: :development
|
63
89
|
prerelease: false
|
64
90
|
version_requirements: !ruby/object:Gem::Requirement
|
65
91
|
requirements:
|
66
92
|
- - "~>"
|
67
93
|
- !ruby/object:Gem::Version
|
68
|
-
version: '
|
94
|
+
version: '2.0'
|
69
95
|
- !ruby/object:Gem::Dependency
|
70
96
|
name: fakeredis
|
71
97
|
requirement: !ruby/object:Gem::Requirement
|
@@ -84,16 +110,16 @@ dependencies:
|
|
84
110
|
name: rake
|
85
111
|
requirement: !ruby/object:Gem::Requirement
|
86
112
|
requirements:
|
87
|
-
- - "
|
113
|
+
- - ">="
|
88
114
|
- !ruby/object:Gem::Version
|
89
|
-
version:
|
115
|
+
version: 12.3.3
|
90
116
|
type: :development
|
91
117
|
prerelease: false
|
92
118
|
version_requirements: !ruby/object:Gem::Requirement
|
93
119
|
requirements:
|
94
|
-
- - "
|
120
|
+
- - ">="
|
95
121
|
- !ruby/object:Gem::Version
|
96
|
-
version:
|
122
|
+
version: 12.3.3
|
97
123
|
- !ruby/object:Gem::Dependency
|
98
124
|
name: rspec
|
99
125
|
requirement: !ruby/object:Gem::Requirement
|
@@ -108,6 +134,20 @@ dependencies:
|
|
108
134
|
- - "~>"
|
109
135
|
- !ruby/object:Gem::Version
|
110
136
|
version: '3.0'
|
137
|
+
- !ruby/object:Gem::Dependency
|
138
|
+
name: appraisal
|
139
|
+
requirement: !ruby/object:Gem::Requirement
|
140
|
+
requirements:
|
141
|
+
- - ">="
|
142
|
+
- !ruby/object:Gem::Version
|
143
|
+
version: '0'
|
144
|
+
type: :development
|
145
|
+
prerelease: false
|
146
|
+
version_requirements: !ruby/object:Gem::Requirement
|
147
|
+
requirements:
|
148
|
+
- - ">="
|
149
|
+
- !ruby/object:Gem::Version
|
150
|
+
version: '0'
|
111
151
|
description:
|
112
152
|
email:
|
113
153
|
- envek@envek.name
|
@@ -115,20 +155,31 @@ executables: []
|
|
115
155
|
extensions: []
|
116
156
|
extra_rdoc_files: []
|
117
157
|
files:
|
158
|
+
- ".github/ISSUE_TEMPLATE/bug_report.md"
|
159
|
+
- ".github/ISSUE_TEMPLATE/feature_request.md"
|
118
160
|
- ".gitignore"
|
119
161
|
- ".rspec"
|
120
162
|
- ".rubocop.yml"
|
121
163
|
- ".travis.yml"
|
164
|
+
- Appraisals
|
165
|
+
- CHANGELOG.md
|
122
166
|
- Gemfile
|
123
167
|
- LICENSE.txt
|
124
168
|
- README.md
|
125
169
|
- Rakefile
|
126
170
|
- bin/console
|
127
171
|
- bin/setup
|
172
|
+
- gemfiles/.bundle/config
|
173
|
+
- gemfiles/anycable_0.6.gemfile
|
174
|
+
- gemfiles/anycable_1.0.gemfile
|
175
|
+
- gemfiles/graphql_1.10.gemfile
|
176
|
+
- gemfiles/graphql_1.11.gemfile
|
177
|
+
- gemfiles/graphql_1.9.gemfile
|
128
178
|
- graphql-anycable.gemspec
|
129
179
|
- lib/Rakefile
|
130
180
|
- lib/graphql-anycable.rb
|
131
181
|
- lib/graphql/anycable.rb
|
182
|
+
- lib/graphql/anycable/cleaner.rb
|
132
183
|
- lib/graphql/anycable/config.rb
|
133
184
|
- lib/graphql/anycable/railtie.rb
|
134
185
|
- lib/graphql/anycable/tasks/clean_expired_subscriptions.rake
|
@@ -153,8 +204,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
153
204
|
- !ruby/object:Gem::Version
|
154
205
|
version: '0'
|
155
206
|
requirements: []
|
156
|
-
|
157
|
-
rubygems_version: 2.7.6
|
207
|
+
rubygems_version: 3.1.2
|
158
208
|
signing_key:
|
159
209
|
specification_version: 4
|
160
210
|
summary: A drop-in replacement for GraphQL ActionCable subscriptions for AnyCable.
|