oath 1.1.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.
Files changed (94) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +7 -0
  3. data/.rspec +1 -0
  4. data/.travis.yml +3 -0
  5. data/Gemfile +3 -0
  6. data/Gemfile.lock +165 -0
  7. data/LICENSE.txt +22 -0
  8. data/NEWS.rdoc +118 -0
  9. data/README.md +384 -0
  10. data/Rakefile +6 -0
  11. data/lib/oath.rb +132 -0
  12. data/lib/oath/back_door.rb +53 -0
  13. data/lib/oath/configuration.rb +149 -0
  14. data/lib/oath/constraints/signed_in.rb +14 -0
  15. data/lib/oath/constraints/signed_out.rb +14 -0
  16. data/lib/oath/controller_helpers.rb +161 -0
  17. data/lib/oath/failure_app.rb +48 -0
  18. data/lib/oath/field_map.rb +56 -0
  19. data/lib/oath/param_transformer.rb +38 -0
  20. data/lib/oath/railtie.rb +11 -0
  21. data/lib/oath/services.rb +5 -0
  22. data/lib/oath/services/authentication.rb +40 -0
  23. data/lib/oath/services/password_reset.rb +27 -0
  24. data/lib/oath/services/sign_in.rb +25 -0
  25. data/lib/oath/services/sign_out.rb +24 -0
  26. data/lib/oath/services/sign_up.rb +42 -0
  27. data/lib/oath/strategies/password_strategy.rb +42 -0
  28. data/lib/oath/test/controller_helpers.rb +43 -0
  29. data/lib/oath/test/helpers.rb +24 -0
  30. data/lib/oath/version.rb +4 -0
  31. data/lib/oath/warden_setup.rb +47 -0
  32. data/oath.gemspec +30 -0
  33. data/spec/features/user/user_signs_in_spec.rb +14 -0
  34. data/spec/features/user/user_signs_in_through_back_door_spec.rb +11 -0
  35. data/spec/features/user/user_tries_to_access_constrained_routes_spec.rb +18 -0
  36. data/spec/features/user/user_tries_to_access_http_auth_page_spec.rb +9 -0
  37. data/spec/features/visitor/visitor_fails_to_sign_up_spec.rb +10 -0
  38. data/spec/features/visitor/visitor_is_unauthorized_spec.rb +8 -0
  39. data/spec/features/visitor/visitor_signs_in_via_invalid_form_spec.rb +11 -0
  40. data/spec/features/visitor/visitor_signs_up_spec.rb +40 -0
  41. data/spec/features/visitor/visitor_tries_to_access_constrained_routes_spec.rb +14 -0
  42. data/spec/features/visitor/visitor_uses_remember_token_spec.rb +13 -0
  43. data/spec/oath/configuration_spec.rb +11 -0
  44. data/spec/oath/controller_helpers_spec.rb +180 -0
  45. data/spec/oath/field_map_spec.rb +19 -0
  46. data/spec/oath/services/authentication_spec.rb +25 -0
  47. data/spec/oath/services/password_reset_spec.rb +24 -0
  48. data/spec/oath/services/sign_in_spec.rb +13 -0
  49. data/spec/oath/services/sign_out_spec.rb +13 -0
  50. data/spec/oath/services/sign_up_spec.rb +49 -0
  51. data/spec/oath/strategies/password_strategy_spec.rb +23 -0
  52. data/spec/oath/test_controller_helpers_spec.rb +63 -0
  53. data/spec/oath/test_helpers_spec.rb +97 -0
  54. data/spec/oath_spec.rb +27 -0
  55. data/spec/rails_app/Rakefile +7 -0
  56. data/spec/rails_app/app/assets/images/rails.png +0 -0
  57. data/spec/rails_app/app/assets/javascripts/application.js +13 -0
  58. data/spec/rails_app/app/assets/stylesheets/application.css +13 -0
  59. data/spec/rails_app/app/controllers/application_controller.rb +4 -0
  60. data/spec/rails_app/app/controllers/basic_auth_controller.rb +7 -0
  61. data/spec/rails_app/app/controllers/constrained_to_users_controller.rb +5 -0
  62. data/spec/rails_app/app/controllers/constrained_to_visitors_controller.rb +5 -0
  63. data/spec/rails_app/app/controllers/failures_controller.rb +5 -0
  64. data/spec/rails_app/app/controllers/invalid_sessions_controller.rb +2 -0
  65. data/spec/rails_app/app/controllers/posts_controller.rb +6 -0
  66. data/spec/rails_app/app/controllers/sessions_controller.rb +26 -0
  67. data/spec/rails_app/app/controllers/users_controller.rb +23 -0
  68. data/spec/rails_app/app/helpers/application_helper.rb +2 -0
  69. data/spec/rails_app/app/models/user.rb +10 -0
  70. data/spec/rails_app/app/views/invalid_sessions/new.html.erb +4 -0
  71. data/spec/rails_app/app/views/layouts/application.html.erb +18 -0
  72. data/spec/rails_app/app/views/posts/index.html.erb +1 -0
  73. data/spec/rails_app/app/views/sessions/new.html.erb +5 -0
  74. data/spec/rails_app/app/views/users/new.html.erb +5 -0
  75. data/spec/rails_app/config.ru +4 -0
  76. data/spec/rails_app/config/application.rb +58 -0
  77. data/spec/rails_app/config/boot.rb +6 -0
  78. data/spec/rails_app/config/database.yml +25 -0
  79. data/spec/rails_app/config/environment.rb +5 -0
  80. data/spec/rails_app/config/environments/development.rb +29 -0
  81. data/spec/rails_app/config/environments/production.rb +54 -0
  82. data/spec/rails_app/config/environments/test.rb +29 -0
  83. data/spec/rails_app/config/initializers/backtrace_silencers.rb +7 -0
  84. data/spec/rails_app/config/initializers/inflections.rb +15 -0
  85. data/spec/rails_app/config/initializers/secret_token.rb +7 -0
  86. data/spec/rails_app/config/routes.rb +24 -0
  87. data/spec/rails_app/db/seeds.rb +7 -0
  88. data/spec/rails_app/public/404.html +26 -0
  89. data/spec/rails_app/public/422.html +26 -0
  90. data/spec/rails_app/public/500.html +25 -0
  91. data/spec/rails_app/public/favicon.ico +0 -0
  92. data/spec/rails_app/script/rails +6 -0
  93. data/spec/spec_helper.rb +37 -0
  94. metadata +325 -0
@@ -0,0 +1,48 @@
1
+ module Oath
2
+ class FailureApp
3
+ def self.call(env)
4
+ request = Rack::Request.new(env)
5
+ new(request).response
6
+ end
7
+
8
+ def initialize(request)
9
+ @request = request
10
+ end
11
+
12
+ def response
13
+ [401, headers, body]
14
+ end
15
+
16
+ private
17
+
18
+ attr_reader :request
19
+
20
+ def headers
21
+ if http_auth_header?
22
+ basic_headers.merge(auth_headers)
23
+ else
24
+ basic_headers
25
+ end
26
+ end
27
+
28
+ def basic_headers
29
+ {
30
+ "Content-Type" => request.content_type.to_s
31
+ }
32
+ end
33
+
34
+ def auth_headers
35
+ {
36
+ "WWW-Authenticate" => 'Basic realm="Application"'
37
+ }
38
+ end
39
+
40
+ def body
41
+ ["Authorization Failed"]
42
+ end
43
+
44
+ def http_auth_header?
45
+ !request.xhr?
46
+ end
47
+ end
48
+ end
@@ -0,0 +1,56 @@
1
+ module Oath
2
+ # FieldMap is used to allow multiple lookup fields. For instance if you
3
+ # wanted to allow a user to sign in via email or username. This is used
4
+ # internally by the authenticate_session controller helper
5
+ # @since 0.0.15
6
+ class FieldMap
7
+ # @param params [Hash] hash of parameters
8
+ # @param field_map [Hash] hash of values to map
9
+ def initialize params, field_map
10
+ @params = params
11
+ @field_map = field_map
12
+ end
13
+
14
+ # converts params into values that can be passed into a where clause
15
+ #
16
+ # @return [Array] if initialized with field_map
17
+ # @return [Hash] if not initialized with field_map
18
+ def to_fields
19
+ if @field_map
20
+ params_from_field_map
21
+ else
22
+ params_with_symbolized_keys
23
+ end
24
+ end
25
+
26
+ private
27
+
28
+ def params_with_symbolized_keys
29
+ @params.inject(default_fields){|hash,(key,value)| hash.merge(key.to_sym => value) }
30
+ end
31
+
32
+ def default_fields
33
+ { Oath.config.user_lookup_field => nil }
34
+ end
35
+
36
+ def params_from_field_map
37
+ [query_string, *([value] * lookup_keys.length)]
38
+ end
39
+
40
+ def query_string
41
+ lookup_keys.map { |key| "#{key} = ?" }.join(" OR ")
42
+ end
43
+
44
+ def session_key
45
+ @field_map.keys.first
46
+ end
47
+
48
+ def lookup_keys
49
+ @field_map.values.first
50
+ end
51
+
52
+ def value
53
+ @params[session_key]
54
+ end
55
+ end
56
+ end
@@ -0,0 +1,38 @@
1
+ module Oath
2
+ # Parameter transformer. Sanitizes and transforms parameter values
3
+ # @since 1.0.0
4
+ class ParamTransformer
5
+ # Initialize parameter transformer
6
+ #
7
+ # @param params [ActionController::Parameters] parameters to be altered
8
+ def initialize(params, transformations)
9
+ @params = params
10
+ @transformations = transformations
11
+ end
12
+
13
+ # Returns the transformed parameters
14
+ def to_h
15
+ sanitized_params.each_with_object({}) do |(key, value), hash|
16
+ hash[key] = transform(key, value)
17
+ end
18
+ end
19
+
20
+ private
21
+
22
+ attr_reader :params, :transformations
23
+
24
+ def sanitized_params
25
+ params.to_h
26
+ end
27
+
28
+ def transform(key, value)
29
+ return value unless value.is_a? String
30
+
31
+ if transformations.key?(key)
32
+ transformations[key].call(value)
33
+ else
34
+ value
35
+ end
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,11 @@
1
+ require 'warden'
2
+
3
+ module Oath
4
+ # Railtie for Oath. Injects the Warden middleware and initializes Oath.
5
+ # @since 0.0.15
6
+ class Railtie < Rails::Railtie
7
+ config.app_middleware.use Warden::Manager do |config|
8
+ Oath.initialize(config)
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,5 @@
1
+ require 'oath/services/sign_in'
2
+ require 'oath/services/sign_out'
3
+ require 'oath/services/sign_up'
4
+ require 'oath/services/authentication'
5
+ require 'oath/services/password_reset'
@@ -0,0 +1,40 @@
1
+ module Oath
2
+ module Services
3
+ # Authentication service. Checks to see if the credentials provided are valid
4
+ # @since 0.0.15
5
+ class Authentication
6
+ # Initialize service
7
+ #
8
+ # @param user [User] A user object
9
+ # @param undigested_token [String] An undigested password
10
+ def initialize user, undigested_token
11
+ @user = user
12
+ @undigested_token = undigested_token
13
+ end
14
+
15
+ # Perform the service
16
+ #
17
+ # @return [User] if authentication succeeds
18
+ # @return [false] if authentication fails
19
+ def perform
20
+ if authenticated?
21
+ user
22
+ else
23
+ false
24
+ end
25
+ end
26
+
27
+ private
28
+
29
+ attr_reader :user, :undigested_token
30
+
31
+ def authenticated?
32
+ user && Oath.compare_token(user.send(token_store_field), undigested_token)
33
+ end
34
+
35
+ def token_store_field
36
+ Oath.config.user_token_store_field
37
+ end
38
+ end
39
+ end
40
+ end
@@ -0,0 +1,27 @@
1
+ module Oath
2
+ module Services
3
+ # Password reset service. Updates the password on a User
4
+ # @since 0.0.15
5
+ class PasswordReset
6
+ # Initialize service
7
+ #
8
+ # @param user [User] A user object
9
+ # @param new_password [String] The new undigested password for a user
10
+ def initialize user, new_password
11
+ @user = user
12
+ @new_password = new_password
13
+ end
14
+
15
+ # Perform the service.
16
+ def perform
17
+ field = Oath.config.user_token_store_field
18
+ digested_password = Oath.hash_token(new_password)
19
+ user[field] = digested_password
20
+ end
21
+
22
+ private
23
+
24
+ attr_reader :user, :new_password
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,25 @@
1
+ module Oath
2
+ module Services
3
+ # Sign in service. Signs the user in via warden
4
+ # @since 0.0.15
5
+ class SignIn
6
+ # Initialize service
7
+ #
8
+ # @param user [User] A user object
9
+ # @param warden [Warden] warden
10
+ def initialize user, warden
11
+ @user = user
12
+ @warden = warden
13
+ end
14
+
15
+ # Perform the service
16
+ def perform
17
+ warden.set_user(user)
18
+ end
19
+
20
+ private
21
+
22
+ attr_reader :warden, :user
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,24 @@
1
+ module Oath
2
+ module Services
3
+ # Sign out service. Signs the user out via warden
4
+ # @since 0.0.15
5
+ class SignOut
6
+ # Initialize service
7
+ #
8
+ # @param warden [Warden] warden
9
+ def initialize warden
10
+ @warden = warden
11
+ @user = warden.user
12
+ end
13
+
14
+ # Perform the service
15
+ def perform
16
+ warden.logout
17
+ end
18
+
19
+ private
20
+
21
+ attr_reader :warden, :user
22
+ end
23
+ end
24
+ end
@@ -0,0 +1,42 @@
1
+ module Oath
2
+ module Services
3
+ # Sign up service. Signs the user up
4
+ # @since 0.0.15
5
+ class SignUp
6
+ # Initialize service
7
+ #
8
+ # @param user_params [Hash] A hash of user credentials. Should contain the lookup and token fields
9
+ def initialize user_params
10
+ digested_token = token_digest(user_params)
11
+ @user_params = user_params.
12
+ except(token_field).
13
+ merge(token_store_field.to_sym => digested_token)
14
+ end
15
+
16
+ # Performs the service
17
+ # @see Oath::Configuration.default_creation_method
18
+ def perform
19
+ Oath.config.creation_method.call(user_params)
20
+ end
21
+
22
+ private
23
+
24
+ attr_reader :user_params
25
+
26
+ def token_digest(user_params)
27
+ undigested_token = user_params[token_field]
28
+ unless undigested_token.blank?
29
+ Oath.hash_token(undigested_token)
30
+ end
31
+ end
32
+
33
+ def token_store_field
34
+ Oath.config.user_token_store_field
35
+ end
36
+
37
+ def token_field
38
+ Oath.config.user_token_field
39
+ end
40
+ end
41
+ end
42
+ end
@@ -0,0 +1,42 @@
1
+ require 'warden'
2
+
3
+ module Oath
4
+ module Strategies
5
+ # Password strategy for warden
6
+ # @since 0.0.15
7
+ class PasswordStrategy < ::Warden::Strategies::Base
8
+
9
+ # Checks if strategy should be executed
10
+ # @return [Boolean]
11
+ def valid?
12
+ lookup_field_value || token_field_value
13
+ end
14
+
15
+
16
+ # Authenticates for warden
17
+ def authenticate!
18
+ user = Oath.config.user_class.find_by(lookup_field => lookup_field_value)
19
+ auth = Oath.config.authentication_service.new(user, token_field_value)
20
+ auth.authenticated? ? success!(user) : fail!("Could not log in")
21
+ end
22
+
23
+ private
24
+
25
+ def lookup_field_value
26
+ params[lookup_field]
27
+ end
28
+
29
+ def token_field_value
30
+ params[token_field]
31
+ end
32
+
33
+ def lookup_field
34
+ Oath.config.user_lookup_field
35
+ end
36
+
37
+ def token_field
38
+ Oath.config.user_token_field
39
+ end
40
+ end
41
+ end
42
+ end
@@ -0,0 +1,43 @@
1
+ require 'warden'
2
+
3
+ module Oath
4
+ module Test
5
+ # These are test helpers for controller specs
6
+ # @note these have only been tested with rspec controller specs
7
+ # @since 0.0.15
8
+ module ControllerHelpers
9
+ def self.included(base)
10
+ base.class_eval do
11
+ setup :store_controller_for_warden, :warden if respond_to?(:setup)
12
+ end
13
+ end
14
+
15
+ # Signs a user in for tests
16
+ # @param user [User] the user to sign in
17
+ def sign_in(user)
18
+ @controller.sign_in(user)
19
+ end
20
+
21
+ # Signs the user out in tests
22
+ def sign_out
23
+ @controller.sign_out
24
+ end
25
+
26
+ # A mock of warden for tests
27
+ def warden
28
+ @warden ||= begin
29
+ manager = Warden::Manager.new(nil) do |config|
30
+ config.merge! Oath.warden_config
31
+ end
32
+ @request.env['warden'] = Warden::Proxy.new(@request.env, manager)
33
+ end
34
+ end
35
+
36
+ private
37
+ def store_controller_for_warden
38
+ @request.env['action_controller.instance'] = @controller
39
+ end
40
+
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,24 @@
1
+ module Oath
2
+ module Test
3
+ # Helpers for integration or feature specs
4
+ # @note these have only been tested with rspec integration and feature specs
5
+ # @since 0.0.15
6
+ module Helpers
7
+ include Warden::Test::Helpers
8
+
9
+ # Sign a user in
10
+ # @param user [User] user to sign in
11
+ # @returns user [User] signed in user
12
+ def sign_in user
13
+ login_as user
14
+
15
+ user
16
+ end
17
+
18
+ # Sign a user out
19
+ def sign_out
20
+ logout
21
+ end
22
+ end
23
+ end
24
+ end
@@ -0,0 +1,4 @@
1
+ module Oath
2
+ # 1.1.0
3
+ VERSION = "1.1.0"
4
+ end