keypairs 0.1.0.alpha.2 → 1.1.1

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: bf4de0021d86f8b939eb09102bf3523ff50c43b65306090823f4d7ea896f85c6
4
- data.tar.gz: 69e0d3738b6e8d3c2d3d1e7b397b08361d83b98f075526ffd9174ab844201109
3
+ metadata.gz: 7d16e91f622673ad1e6b0675d86fefecfcf2dea5f49d04e4e412eb6d13b1690e
4
+ data.tar.gz: fce2c5965a3487437f6f265244bcc9eff7f04435104826fd9d48df51a8a7425e
5
5
  SHA512:
6
- metadata.gz: 0e2acb9c0ec8069c3de1885bb810d6d759fa5e6594ce309feffb3abd5e357338106740949185734731f2fd9368b06472cbc2cd2dedc2a283711c78a17587c6af
7
- data.tar.gz: 359aeb886e535c34f4c8071b4043e1186c92e91947b05c6215cd45efd3a54314abb85be17d8e56b5e186581b8a166f9b8f5f2e22ab4cc1af8a9590d796c9c546
6
+ metadata.gz: 608c44fa803ecd3af3bcd2e6bc16feaddefa073032c8b569d56756c23814e8bd035b2453e561de333786abc8b3c32a99d1b824e0a48d050afa8853103b255e7e
7
+ data.tar.gz: 586339052532023f863e8fa0fe1ab3ddce3636271be7375a21af04c3f01d348c8bdead9e6e03cd9a7d5cec0aecad40e3fa4078e1907ecd122846a90940e7109f
data/README.md CHANGED
@@ -4,15 +4,28 @@ Applications often need to have a public/private keypair so sign messages. This
4
4
  Note: This gem is intended to work within Rails applications. It can probably be adjusted easily to also work for non-rails / sinatra project but that's out of scope for now.
5
5
 
6
6
  ## Installation
7
- Add this line to your application's Gemfile:
7
+ 1. **Add gem**
8
8
 
9
- ```ruby
10
- gem 'keypairs'
11
- ```
9
+ Add this line to your application's Gemfile:
10
+
11
+ ```ruby
12
+ gem 'keypairs'
13
+ ```
14
+
15
+ The of course run `bundle install`.
16
+
17
+ 2. **Copy migration**
18
+
19
+ The default migration file can be copied to your app with:
20
+ ```bash
21
+ bundle exec rails keypairs:install:migrations
22
+ ```
23
+
24
+ Then of course run `bundle exec rails db:migrate`
12
25
 
13
- The of course run `bundle install` and run the migrations `bundle exec rake db:migrate`. The migrations from the gem run automatically.
26
+ 3. **Setup encryption key**
14
27
 
15
- The private keys are encrypted with the [lockbox](https://github.com/ankane/lockbox) gem. In order for this to work, you need to set the master key as described in [the readme](https://github.com/ankane/lockbox#key-generation), but the easiest thing is to set the environment variable `LOCKBOX_MASTER_KEY` to a sufficient long string (you can generate one with `Lockbox.generate_key`).
28
+ The private keys are encrypted with the [lockbox](https://github.com/ankane/lockbox) gem. In order for this to work, you need to set the master key as described in [the readme](https://github.com/ankane/lockbox#key-generation), but the easiest thing is to set the environment variable `LOCKBOX_MASTER_KEY` to a sufficient long string (you can generate one with `Lockbox.generate_key`).
16
29
 
17
30
  ## Usage
18
31
  The central point of this gem is the `Keypair` model which is backed by the `keypairs` table. If you need to sign messages, you can get the current keypair with the `Keypair.current` method. This method performs the rotation of the keypairs if required.
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ class AddValidityToKeypairs < ActiveRecord::Migration[6.0]
4
+ class Keypair < ActiveRecord::Base; end
5
+
6
+ def change # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
7
+ change_table :keypairs, bulk: true do |t|
8
+ t.datetime :not_before, precision: 6
9
+ t.datetime :not_after, precision: 6
10
+ t.datetime :expires_at, precision: 6
11
+ end
12
+
13
+ reversible do |dir|
14
+ dir.up do
15
+ Keypair.find_each do |keypair|
16
+ keypair.update!(
17
+ not_before: keypair.created_at,
18
+ not_after: keypair.created_at + 1.month,
19
+ expires_at: keypair.created_at + 2.months
20
+ )
21
+ end
22
+ end
23
+ end
24
+
25
+ change_column_null :keypairs, :not_before, false
26
+ change_column_null :keypairs, :not_after, false
27
+ change_column_null :keypairs, :expires_at, false
28
+ end
29
+ end
@@ -6,8 +6,18 @@ require 'jwt'
6
6
  # This class contains functionality needed for signing messages
7
7
  # and publishing JWK[s].
8
8
  #
9
- # The last three created keypairs are considered valid, so creating a new Keypair
10
- # will invalidate the second to last created Keypair.
9
+ # Keypairs are considered valid based on their {#not_before}, {#not_after} and {#expires_at} attributes.
10
+ #
11
+ # A keypair can be used for signing if:
12
+ # - The current time is greater than or equal to {#not_before}
13
+ # - The current time is less than or equal to {#not_after}
14
+ #
15
+ # A keypair can be used for validation if:
16
+ # - The current time is less than {#expires_at}.
17
+ #
18
+ # By default, this means that when a key is created, it can be used for signing for 1 month and can still be used
19
+ # for signature validation 1 month after it is not used for signing (i.e. for 2 months since it started being used
20
+ # for signing).
11
21
  #
12
22
  # If you need to sign messages, use the {Keypair.current} keypair for this. This method
13
23
  # performs the rotation of the keypairs if required.
@@ -21,26 +31,36 @@ require 'jwt'
21
31
  # decoded = Keypair.jwt_decode(id_token)
22
32
  #
23
33
  # @attr [String] jwk_kid The public external id of the key used to find the associated key on decoding.
34
+ # @attr [Time] not_before The time before which no payloads may be signed using the keypair.
35
+ # @attr [Time] not_after The time after which no payloads may be signed using the keypair.
36
+ # @attr [Time] expires_at The time after which the keypair may not be used for signature validation.
24
37
  class Keypair < ActiveRecord::Base
25
38
  ALGORITHM = 'RS256'
26
39
  ROTATION_INTERVAL = 1.month
27
40
 
28
- encrypts :_keypair
41
+ lockbox_encrypts :_keypair
29
42
 
30
43
  validates :_keypair, presence: true
31
44
  validates :jwk_kid, presence: true
45
+ validates :not_before, :expires_at, presence: true
46
+
47
+ validate :not_after_after_not_before
48
+ validate :expires_at_after_not_after
32
49
 
33
50
  after_initialize :set_keypair
51
+ after_initialize :set_validity
34
52
 
35
53
  # @!method valid
36
54
  # @!scope class
37
- # The last 3 keypairs are considered valid and can be used to validate signatures and export public jwks.
38
- # It uses a subquery to make sure a +find_by+ actually searches only the valid 3 ones.
39
- scope :valid, -> { where(id: unscoped.order(created_at: :desc).limit(3)) }
55
+ # Non-expired keypairs are considered valid and can be used to validate signatures and export public jwks.
56
+ scope :valid, -> { where(arel_table[:expires_at].gt(Time.zone.now)) }
40
57
 
41
- # @return [Keypair] the keypair used to sign messages and autorotates if it is older than 1 month.
58
+ # @return [Keypair] the keypair used to sign messages and autorotates if it has expired.
42
59
  def self.current
43
- order(:created_at).where(arel_table[:created_at].gt(1.month.ago)).last || create!
60
+ order(not_before: :asc)
61
+ .where(arel_table[:not_before].lteq(Time.zone.now))
62
+ .where(arel_table[:not_after].gteq(Time.zone.now))
63
+ .last || create!
44
64
  end
45
65
 
46
66
  # The JWK Set of our valid keypairs.
@@ -66,11 +86,26 @@ class Keypair < ActiveRecord::Base
66
86
  #
67
87
  # @see https://www.imsglobal.org/spec/security/v1p0/#h_key-set-url
68
88
  def self.keyset
89
+ valid_keys = valid.order(not_before: :asc).to_a
90
+ # If we don't have any keys or if we don't have a future key (i.e. the last key is the current key)
91
+ while valid_keys.last.nil? || valid_keys.last.not_before <= Time.zone.now
92
+ # There is an automatic fallback to Time.zone.now if not_before is not set
93
+ valid_keys << create!(not_before: valid_keys.last&.not_after)
94
+ end
95
+
69
96
  {
70
- keys: valid.order(created_at: :desc).map(&:public_jwk_export)
97
+ keys: valid_keys.map(&:public_jwk_export)
71
98
  }
72
99
  end
73
100
 
101
+ # @return [Hash] a cached version of the keyset
102
+ # @see #keyset
103
+ def self.cached_keyset
104
+ Rails.cache.fetch('keypairs/Keypair/keyset', expires_in: 12.hours) do
105
+ keyset
106
+ end
107
+ end
108
+
74
109
  # Encodes the payload with the current keypair.
75
110
  # It forewards the call to the instance method {Keypair#jwt_encode}.
76
111
  # @return [String] Encoded JWT token with security credentials.
@@ -173,4 +208,25 @@ class Keypair < ActiveRecord::Base
173
208
  self._keypair ||= OpenSSL::PKey::RSA.new(2048).to_pem
174
209
  self.jwk_kid = public_jwk.kid
175
210
  end
211
+
212
+ # Set the validity timestamps based on the rotation interval.
213
+ def set_validity
214
+ self.not_before ||= created_at || Time.zone.now
215
+ self.not_after ||= not_before + ROTATION_INTERVAL
216
+ self.expires_at ||= not_after + ROTATION_INTERVAL
217
+ end
218
+
219
+ def not_after_after_not_before
220
+ return if not_before.nil? || not_after.nil?
221
+ return if not_after > not_before
222
+
223
+ errors.add(:not_after, 'must be after not before')
224
+ end
225
+
226
+ def expires_at_after_not_after
227
+ return if not_after.nil? || expires_at.nil?
228
+ return if expires_at > not_after
229
+
230
+ errors.add(:expires_at, 'must be after not after')
231
+ end
176
232
  end
@@ -1,18 +1,8 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Keypairs
4
- # Rails engine for this gem.
5
- # It ensures that the migrations are automatically ran in the applications.
4
+ # This engine is only needed to add the migration installation rake task.
6
5
  class Engine < ::Rails::Engine
7
- initializer :append_migrations do |app|
8
- unless app.root.to_s.match? "#{root}/"
9
- config.paths['db/migrate'].expanded.each do |expanded_path|
10
- app.config.paths['db/migrate'] << expanded_path
11
- end
12
- # Apartment will modify this, but it doesn't fully support engine migrations,
13
- # so we'll reset it here
14
- ActiveRecord::Migrator.migrations_paths = app.paths['db/migrate'].to_a
15
- end
16
- end
6
+ engine_name 'keypairs'
17
7
  end
18
8
  end
@@ -18,7 +18,9 @@ module Keypairs
18
18
  # }
19
19
  class PublicKeysController < ActionController::API
20
20
  def index
21
- render json: Keypair.keyset
21
+ # Always cache for 1 week, our rotation interval is much more than a week
22
+ expires_in 1.week, public: true
23
+ render json: Keypair.cached_keyset
22
24
  end
23
25
  end
24
26
  end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Keypairs
4
- VERSION = '0.1.0.alpha.2'
4
+ VERSION = '1.1.1'
5
5
  end
data/lib/keypairs.rb CHANGED
@@ -1,4 +1,12 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require 'lockbox'
4
- require 'keypairs/engine'
4
+
5
+ autoload :Keypair, 'keypair.rb'
6
+
7
+ # The Keypairs module contains common functionality in support of the {Keypair} model.
8
+ module Keypairs
9
+ autoload :PublicKeysController, 'keypairs/public_keys_controller'
10
+ end
11
+
12
+ require 'keypairs/engine' if defined?(Rails)
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: keypairs
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.0.alpha.2
4
+ version: 1.1.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Stef Schenkelaars
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2020-10-26 00:00:00.000000000 Z
11
+ date: 2022-02-22 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: actionpack
@@ -66,6 +66,20 @@ dependencies:
66
66
  - - "~>"
67
67
  - !ruby/object:Gem::Version
68
68
  version: '0.4'
69
+ - !ruby/object:Gem::Dependency
70
+ name: appraisal
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ">="
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
69
83
  - !ruby/object:Gem::Dependency
70
84
  name: brakeman
71
85
  requirement: !ruby/object:Gem::Requirement
@@ -206,6 +220,20 @@ dependencies:
206
220
  - - ">="
207
221
  - !ruby/object:Gem::Version
208
222
  version: '0'
223
+ - !ruby/object:Gem::Dependency
224
+ name: timecop
225
+ requirement: !ruby/object:Gem::Requirement
226
+ requirements:
227
+ - - ">="
228
+ - !ruby/object:Gem::Version
229
+ version: '0'
230
+ type: :development
231
+ prerelease: false
232
+ version_requirements: !ruby/object:Gem::Requirement
233
+ requirements:
234
+ - - ">="
235
+ - !ruby/object:Gem::Version
236
+ version: '0'
209
237
  description: Manage application level keypairs with automatic rotation and JWT support
210
238
  email:
211
239
  - stef.schenkelaars@gmail.com
@@ -215,11 +243,12 @@ extra_rdoc_files: []
215
243
  files:
216
244
  - LICENSE
217
245
  - README.md
218
- - app/controllers/keypairs/public_keys_controller.rb
219
- - app/models/keypair.rb
220
246
  - db/migrate/20201024100500_create_keypairs.rb
247
+ - db/migrate/20201116144600_add_validity_to_keypairs.rb
248
+ - lib/keypair.rb
221
249
  - lib/keypairs.rb
222
250
  - lib/keypairs/engine.rb
251
+ - lib/keypairs/public_keys_controller.rb
223
252
  - lib/keypairs/version.rb
224
253
  homepage: https://drieam.github.io/keypairs
225
254
  licenses:
@@ -235,14 +264,14 @@ required_ruby_version: !ruby/object:Gem::Requirement
235
264
  requirements:
236
265
  - - ">="
237
266
  - !ruby/object:Gem::Version
238
- version: 2.5.0
267
+ version: 2.7.0
239
268
  required_rubygems_version: !ruby/object:Gem::Requirement
240
269
  requirements:
241
- - - ">"
270
+ - - ">="
242
271
  - !ruby/object:Gem::Version
243
- version: 1.3.1
272
+ version: '0'
244
273
  requirements: []
245
- rubygems_version: 3.1.4
274
+ rubygems_version: 3.3.7
246
275
  signing_key:
247
276
  specification_version: 4
248
277
  summary: Manage application level keypairs with automatic rotation and JWT support