graphql-anycable 0.3.2 → 0.5.0

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 79ed158f4e882659ca963af6a22fb66ce34dfa9a2e0f07de40f6331dd5865fb3
4
- data.tar.gz: 47a9a999c879d9a143d48fea3275573f557e0b33615038b0b6940c7743338939
3
+ metadata.gz: 6c831aba1f67077cccb4ef5f1b7858715cf056e554551dab59acdf265a8c671e
4
+ data.tar.gz: 5049b208e8a02a2e110e698f5336a1314577d2e1f8a37742ce07807daac6e41a
5
5
  SHA512:
6
- metadata.gz: a960f6150eb79e81fe27c12a12d8fee66297aaac03184725df45f1c95abdc0274e211564cd4095c5e64a88ee8694ee7972f071834279022f1b779be9580ba7ec
7
- data.tar.gz: e98df2b3bb0770a0531b982002a319637821cc842be29ea832e916ffafbaf827ec1d9c275604ce13ecdcb278213d8bc9b54013981406f0d23ef95399bb552001
6
+ metadata.gz: d9451177ced38d8f963fe8b7f4c5b8ae1933a1d9730403202ee0864f23b3649752e538f8e480aedcae042aac2958b60145dc1305b79e67a3e480a96279df2bb1
7
+ data.tar.gz: 1367ac8df59b0470725197da7be2701175cb8571146f81eaa896dd00b3052eee1f91e2abdabdf67bb5a18754385e11169d942864a65e1fb59cdb7f8afdd5938e
@@ -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.
@@ -0,0 +1,64 @@
1
+ name: Build and release gem to RubyGems
2
+
3
+ on:
4
+ push:
5
+ tags:
6
+ - v*
7
+
8
+ jobs:
9
+ release:
10
+ runs-on: ubuntu-latest
11
+ steps:
12
+ - uses: actions/checkout@v2
13
+ with:
14
+ fetch-depth: 0 # Fetch current tag as annotated. See https://github.com/actions/checkout/issues/290
15
+ - uses: ruby/setup-ruby@v1
16
+ with:
17
+ ruby-version: 2.7
18
+ - name: "Extract data from tag: version, message, body"
19
+ id: tag
20
+ run: |
21
+ git fetch --tags --force # Really fetch annotated tag. See https://github.com/actions/checkout/issues/290#issuecomment-680260080
22
+ echo ::set-output name=version::${GITHUB_REF#refs/tags/v}
23
+ echo ::set-output name=subject::$(git for-each-ref $GITHUB_REF --format='%(contents:subject)')
24
+ # Multiline body for release. See https://github.community/t/set-output-truncates-multiline-strings/16852/5
25
+ BODY="$(git for-each-ref $GITHUB_REF --format='%(contents:body)')"
26
+ BODY="${BODY//'%'/'%25'}"
27
+ BODY="${BODY//$'\n'/'%0A'}"
28
+ BODY="${BODY//$'\r'/'%0D'}"
29
+ echo "::set-output name=body::$BODY"
30
+ - name: Build gem
31
+ run: gem build
32
+ - name: Check version
33
+ run: ls graphql-anycable-${{ steps.tag.outputs.version }}.gem
34
+ - name: Create Release
35
+ id: create_release
36
+ uses: actions/create-release@v1
37
+ env:
38
+ GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
39
+ with:
40
+ tag_name: ${{ github.ref }}
41
+ release_name: ${{ steps.tag.outputs.subject }}
42
+ body: ${{ steps.tag.outputs.body }}
43
+ draft: false
44
+ prerelease: false
45
+ - name: Upload Release Asset
46
+ id: upload-release-asset
47
+ uses: actions/upload-release-asset@v1
48
+ env:
49
+ GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
50
+ with:
51
+ upload_url: ${{ steps.create_release.outputs.upload_url }}
52
+ asset_path: graphql-anycable-${{ steps.tag.outputs.version }}.gem
53
+ asset_name: graphql-anycable-${{ steps.tag.outputs.version }}.gem
54
+ asset_content_type: application/x-tar
55
+ - name: Install publish prerequisites
56
+ run: |
57
+ sudo apt-get update
58
+ sudo apt-get install oathtool
59
+ - name: Publish to RubyGems
60
+ env:
61
+ GEM_HOST_API_KEY: "${{ secrets.RUBYGEMS_API_KEY }}"
62
+ RUBYGEMS_OTP_KEY: "${{ secrets.RUBYGEMS_OTP_KEY }}"
63
+ run: |
64
+ gem push graphql-anycable-${{ steps.tag.outputs.version }}.gem --otp=$(oathtool --totp --base32 $RUBYGEMS_OTP_KEY)
@@ -0,0 +1,69 @@
1
+ name: Run tests
2
+
3
+ on:
4
+ pull_request:
5
+ push:
6
+ branches:
7
+ - '**'
8
+ tags-ignore:
9
+ - 'v*'
10
+
11
+ jobs:
12
+ test:
13
+ name: "Run tests"
14
+ if: "! contains(toJSON(github.event.commits.latest.message), '[ci skip]')"
15
+ runs-on: ubuntu-latest
16
+ strategy:
17
+ fail-fast: false
18
+ matrix:
19
+ include:
20
+ - ruby: 2.7
21
+ graphql: '~> 1.11.0'
22
+ anycable: '~> 1.0.0'
23
+ interpreter: yes
24
+ - ruby: 2.7
25
+ graphql: '~> 1.11.0'
26
+ anycable: '~> 1.0.0'
27
+ interpreter: no
28
+ - ruby: 2.6
29
+ graphql: '~> 1.10.0'
30
+ anycable: '~> 1.0.0'
31
+ interpreter: yes
32
+ - ruby: 2.6
33
+ graphql: '~> 1.10.0'
34
+ anycable: '~> 1.0.0'
35
+ interpreter: no
36
+ - ruby: 2.5
37
+ graphql: '~> 1.9.0'
38
+ anycable: '~> 0.6.0'
39
+ interpreter: yes
40
+ - ruby: 2.5
41
+ graphql: '~> 1.9.0'
42
+ anycable: '~> 0.6.0'
43
+ interpreter: no
44
+ container:
45
+ image: ruby:${{ matrix.ruby }}
46
+ env:
47
+ CI: true
48
+ GRAPHQL_RUBY_VERSION: ${{ matrix.graphql }}
49
+ ANYCABLE_VERSION: ${{ matrix.anycable }}
50
+ steps:
51
+ - uses: actions/checkout@v2
52
+ - uses: actions/cache@v2
53
+ with:
54
+ path: vendor/bundle
55
+ key: bundle-${{ matrix.ruby }}-${{ matrix.graphql }}-${{ matrix.anycable }}-${{ hashFiles('**/*.gemspec') }}-${{ hashFiles('**/Gemfile') }}
56
+ restore-keys: |
57
+ bundle-${{ matrix.ruby }}-${{ matrix.graphql }}-${{ matrix.anycable }}-
58
+ bundle-${{ matrix.ruby }}-
59
+ - name: Upgrade Bundler to 2.0 (for older Rubies)
60
+ run: gem install bundler -v '~> 2.0'
61
+ - name: Bundle install
62
+ run: |
63
+ bundle config path vendor/bundle
64
+ bundle install
65
+ bundle update
66
+ - name: Run RSpec
67
+ env:
68
+ GRAPHQL_RUBY_INTERPRETER: ${{ matrix.interpreter }}
69
+ run: bundle exec rspec
data/.gitignore CHANGED
@@ -8,5 +8,4 @@
8
8
  /tmp/
9
9
  Gemfile.lock
10
10
  .rspec_status
11
-
12
-
11
+ /gemfiles/*.lock
@@ -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
@@ -7,10 +7,16 @@ git_source(:github) { |repo_name| "https://github.com/#{repo_name}" }
7
7
  # Specify your gem's dependencies in graphql-anycable.gemspec
8
8
  gemspec
9
9
 
10
+ gem "graphql", ENV.fetch("GRAPHQL_RUBY_VERSION", "~> 1.11")
11
+ gem "anycable", ENV.fetch("ANYCABLE_VERSION", "~> 1.0")
12
+
10
13
  group :development, :test do
11
14
  gem "pry"
12
15
  gem "pry-byebug", platform: :mri
13
16
 
14
17
  gem "rubocop"
15
18
  gem "rubocop-rspec"
19
+
20
+ # See https://github.com/guilleiguaran/fakeredis/pull/247
21
+ gem "fakeredis", github: 'guilleiguaran/fakeredis'
16
22
  end
data/README.md CHANGED
@@ -50,7 +50,7 @@ Or install it yourself as:
50
50
 
51
51
  ```ruby
52
52
  class MySchema < GraphQL::Schema
53
- use GraphQL::Subscriptions::AnyCableSubscriptions
53
+ use GraphQL::AnyCable
54
54
 
55
55
  subscription SubscriptionType
56
56
  end
@@ -103,11 +103,78 @@ Or install it yourself as:
103
103
 
104
104
  To avoid filling Redis storage with stale subscription data:
105
105
 
106
- 1. Set `GRAPHQL_ANYCABLE_SUBSCRIPTION_EXPIRATION_SECONDS` environment variable to number of seconds (e.g. `604800` for 1 week). See [anyway_config] documentation to other ways of configuring this gem.
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
107
 
108
108
  2. Execute `rake graphql:anycable:clean` once in a while to clean up stale subscription data.
109
109
 
110
- Heroku users should set up `GRAPHQL_ANYCABLE_USE_REDIS_OBJECT_ON_CLEANUP` environment variable to `false` due to [limitations in Heroku Redis](https://devcenter.heroku.com/articles/heroku-redis#connection-permissions).
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!
141
+
142
+ ## Data model
143
+
144
+ As in AnyCable there is no place to store subscription data in-memory, it should be persisted somewhere to be retrieved on `GraphQLSchema.subscriptions.trigger` and sent to subscribed clients. `graphql-anycable` uses the same Redis database as AnyCable itself.
145
+
146
+ 1. Event subscriptions: `graphql-event:#{event.topic}` set containing identifiers for all subscriptions for given operation with certain context and arguments (serialized in _topic_). Used to find all subscriptions on `GraphQLSchema.subscriptions.trigger`.
147
+
148
+ ```
149
+ SMEMBERS graphql-event:1:myStats:
150
+ => 52ee8d65-275e-4d22-94af-313129116388
151
+ ```
152
+
153
+ 2. Subscription data: `graphql-subscription:#{subscription_id}` hash contains everything required to evaluate subscription on trigger and create data for client.
154
+
155
+ ```
156
+ HGETALL graphql-subscription:52ee8d65-275e-4d22-94af-313129116388
157
+ => {
158
+ context: '{"user_id":1,"user":{"__gid__":"Z2lkOi8vZWJheS1tYWcyL1VzZXIvMQ"},"subscription_id":"52ee8d65-275e-4d22-94af-313129116388\","action_cable_stream":"graphql-subscription:52ee8d65-275e-4d22-94af-313129116388",}',
159
+ variables: '{}',
160
+ operation_name: 'MyStats'
161
+ query_string: 'subscription MyStats { myStatsUpdated { completed total processed __typename } }',
162
+ }
163
+ ```
164
+
165
+ 3. Channel subscriptions: `graphql-channel:#{channel_id}` set containing identifiers for subscriptions created in ActionCable channel to delete them on client disconnect.
166
+
167
+ ```
168
+ SMEMBERS graphql-channel:17420c6ed9e
169
+ => 52ee8d65-275e-4d22-94af-313129116388
170
+ ```
171
+
172
+ 4. Subscription events: `graphql-subscription-events:#{subscription_id}` set containing event topics to delete subscription identifier from event subscriptions set on unsubscribe (or client disconnect).
173
+
174
+ ```
175
+ SMEMBERS graphql-subscription-events:52ee8d65-275e-4d22-94af-313129116388
176
+ => 1:myStats:
177
+ ```
111
178
 
112
179
  ## Development
113
180
 
@@ -6,7 +6,7 @@ require "graphql/anycable/version"
6
6
 
7
7
  Gem::Specification.new do |spec|
8
8
  spec.name = "graphql-anycable"
9
- spec.version = GraphQL::Anycable::VERSION
9
+ spec.version = GraphQL::AnyCable::VERSION
10
10
  spec.authors = ["Andrey Novikov"]
11
11
  spec.email = ["envek@envek.name"]
12
12
 
@@ -29,6 +29,7 @@ Gem::Specification.new do |spec|
29
29
  spec.add_dependency "anycable", ">= 0.6.0", "< 2"
30
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
34
  spec.add_development_dependency "bundler", "~> 2.0"
34
35
  spec.add_development_dependency "fakeredis"
@@ -3,23 +3,39 @@
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"
9
10
 
10
11
  module GraphQL
11
- module Anycable
12
+ module AnyCable
13
+ def self.use(schema)
14
+ schema.use GraphQL::Subscriptions::AnyCableSubscriptions
15
+ end
16
+
12
17
  module_function
13
18
 
14
19
  def redis
15
20
  @redis ||= begin
16
- adapter = ::Anycable.broadcast_adapter
21
+ adapter = ::AnyCable.broadcast_adapter
17
22
  unless adapter.is_a?(::AnyCable::BroadcastAdapters::Redis)
18
23
  raise "Unsupported AnyCable adapter: #{adapter.class}. " \
19
24
  "graphql-anycable works only with redis broadcast adapter."
20
25
  end
21
- ::Anycable.broadcast_adapter.redis_conn
26
+ ::AnyCable.broadcast_adapter.redis_conn
22
27
  end
23
28
  end
29
+
30
+ def config
31
+ @config ||= GraphQL::AnyCable::Config.new
32
+ end
33
+
34
+ def configure
35
+ yield(config) if block_given?
36
+ end
24
37
  end
25
38
  end
39
+
40
+ # For backward compatibility
41
+ GraphQL::Anycable = GraphQL::AnyCable
@@ -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
@@ -3,7 +3,7 @@
3
3
  require "anyway"
4
4
 
5
5
  module GraphQL
6
- module Anycable
6
+ module AnyCable
7
7
  class Config < Anyway::Config
8
8
  config_name :graphql_anycable
9
9
  env_prefix :graphql_anycable
@@ -3,7 +3,7 @@
3
3
  require "rails"
4
4
 
5
5
  module GraphQL
6
- module Anycable
6
+ module AnyCable
7
7
  class Railtie < ::Rails::Railtie
8
8
  rake_tasks do
9
9
  path = File.expand_path(__dir__)
@@ -1,7 +1,7 @@
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
7
  desc "Clean up stale graphql channels, subscriptions, and events from redis"
@@ -11,54 +11,20 @@ namespace :graphql do
11
11
  task clean_expired_subscriptions: :clean
12
12
 
13
13
  namespace :clean do
14
- KLASS = GraphQL::Subscriptions::AnyCableSubscriptions
15
-
16
14
  # Clean up old channels
17
15
  task :channels do
18
- next unless config.subscription_expiration_seconds
19
- next unless config.use_redis_object_on_cleanup
20
-
21
- redis.scan_each(match: "#{KLASS::CHANNEL_PREFIX}*") do |key|
22
- idle = redis.object("IDLETIME", key)
23
- next if idle&.<= config.subscription_expiration_seconds
24
-
25
- redis.del(key)
26
- end
16
+ GraphQL::AnyCable::Cleaner.clean_channels
27
17
  end
28
18
 
29
19
  # Clean up old subscriptions (they should have expired by themselves)
30
20
  task :subscriptions do
31
- next unless config.subscription_expiration_seconds
32
- next unless config.use_redis_object_on_cleanup
33
-
34
- redis.scan_each(match: "#{KLASS::SUBSCRIPTION_PREFIX}*") do |key|
35
- idle = redis.object("IDLETIME", key)
36
- next if idle&.<= config.subscription_expiration_seconds
37
-
38
- redis.del(key)
39
- end
21
+ GraphQL::AnyCable::Cleaner.clean_subscriptions
40
22
  end
41
23
 
42
24
  # Clean up subscription_ids from events for expired subscriptions
43
25
  task :events do
44
- redis.scan_each(match: "#{KLASS::SUBSCRIPTION_EVENTS_PREFIX}*") do |key|
45
- subscription_id = key.sub(/\A#{KLASS::SUBSCRIPTION_EVENTS_PREFIX}/, "")
46
- next if redis.exists(KLASS::SUBSCRIPTION_PREFIX + subscription_id)
47
-
48
- redis.smembers(key).each do |event_topic|
49
- redis.srem(KLASS::EVENT_PREFIX + event_topic, subscription_id)
50
- end
51
- redis.del(key)
52
- end
26
+ GraphQL::AnyCable::Cleaner.clean_events
53
27
  end
54
28
  end
55
-
56
- def config
57
- @config ||= GraphQL::Anycable::Config.new
58
- end
59
-
60
- def redis
61
- GraphQL::Anycable.redis
62
- end
63
29
  end
64
30
  end
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module GraphQL
4
- module Anycable
5
- VERSION = "0.3.2"
4
+ module AnyCable
5
+ VERSION = "0.5.0"
6
6
  end
7
7
  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 = { result: result.to_h, more: true }
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, *events.map(&:topic))
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,9 +115,9 @@ 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] = @serializer.load(subscription[:context])
120
+ subscription[:context] = @serializer.load(subscription[:context])
121
121
  subscription[:variables] = JSON.parse(subscription[:variables])
122
122
  subscription[:operation_name] = nil if subscription[:operation_name].strip == ""
123
123
  end
@@ -145,11 +145,7 @@ module GraphQL
145
145
  private
146
146
 
147
147
  def anycable
148
- @anycable ||= ::Anycable.broadcast_adapter
149
- end
150
-
151
- def config
152
- @config ||= GraphQL::Anycable::Config.new
148
+ @anycable ||= ::AnyCable.broadcast_adapter
153
149
  end
154
150
  end
155
151
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: graphql-anycable
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.3.2
4
+ version: 0.5.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Andrey Novikov
8
8
  autorequire:
9
9
  bindir: exe
10
10
  cert_chain: []
11
- date: 2020-03-02 00:00:00.000000000 Z
11
+ date: 2020-08-25 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: anycable
@@ -64,6 +64,20 @@ dependencies:
64
64
  - - "~>"
65
65
  - !ruby/object:Gem::Version
66
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
67
81
  - !ruby/object:Gem::Dependency
68
82
  name: bundler
69
83
  requirement: !ruby/object:Gem::Requirement
@@ -127,10 +141,14 @@ executables: []
127
141
  extensions: []
128
142
  extra_rdoc_files: []
129
143
  files:
144
+ - ".github/ISSUE_TEMPLATE/bug_report.md"
145
+ - ".github/ISSUE_TEMPLATE/feature_request.md"
146
+ - ".github/workflows/build-release.yml"
147
+ - ".github/workflows/test.yml"
130
148
  - ".gitignore"
131
149
  - ".rspec"
132
150
  - ".rubocop.yml"
133
- - ".travis.yml"
151
+ - CHANGELOG.md
134
152
  - Gemfile
135
153
  - LICENSE.txt
136
154
  - README.md
@@ -141,6 +159,7 @@ files:
141
159
  - lib/Rakefile
142
160
  - lib/graphql-anycable.rb
143
161
  - lib/graphql/anycable.rb
162
+ - lib/graphql/anycable/cleaner.rb
144
163
  - lib/graphql/anycable/config.rb
145
164
  - lib/graphql/anycable/railtie.rb
146
165
  - lib/graphql/anycable/tasks/clean_expired_subscriptions.rake
@@ -165,7 +184,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
165
184
  - !ruby/object:Gem::Version
166
185
  version: '0'
167
186
  requirements: []
168
- rubygems_version: 3.0.3
187
+ rubygems_version: 3.1.2
169
188
  signing_key:
170
189
  specification_version: 4
171
190
  summary: A drop-in replacement for GraphQL ActionCable subscriptions for AnyCable.
@@ -1,10 +0,0 @@
1
- ---
2
- sudo: false
3
- language: ruby
4
- cache: bundler
5
- rvm:
6
- - 2.7.0
7
- - 2.6.5
8
- - 2.5.7
9
- - 2.4.9
10
- before_install: gem install bundler -v "~> 2.0"