rails_api_auth 0.0.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (65) hide show
  1. checksums.yaml +7 -0
  2. data/LICENSE +22 -0
  3. data/README.md +3 -0
  4. data/Rakefile +20 -0
  5. data/app/controllers/oauth2_controller.rb +54 -0
  6. data/app/controllers/rails_api_auth/application_controller.rb +7 -0
  7. data/app/lib/login_not_found.rb +9 -0
  8. data/app/models/login.rb +47 -0
  9. data/app/services/facebook_authenticator.rb +64 -0
  10. data/config/initializers/facebook.rb +6 -0
  11. data/config/routes.rb +4 -0
  12. data/db/migrate/20150709221755_create_logins.rb +16 -0
  13. data/lib/rails_api_auth.rb +5 -0
  14. data/lib/rails_api_auth/authentication.rb +32 -0
  15. data/lib/rails_api_auth/engine.rb +19 -0
  16. data/lib/rails_api_auth/version.rb +5 -0
  17. data/lib/tasks/rails_api_auth_tasks.rake +4 -0
  18. data/spec/dummy/README.rdoc +28 -0
  19. data/spec/dummy/Rakefile +6 -0
  20. data/spec/dummy/app/assets/javascripts/application.js +13 -0
  21. data/spec/dummy/app/assets/stylesheets/application.css +15 -0
  22. data/spec/dummy/app/controllers/application_controller.rb +7 -0
  23. data/spec/dummy/app/controllers/authenticated_controller.rb +13 -0
  24. data/spec/dummy/bin/bundle +3 -0
  25. data/spec/dummy/bin/rails +4 -0
  26. data/spec/dummy/bin/rake +4 -0
  27. data/spec/dummy/bin/setup +29 -0
  28. data/spec/dummy/config.ru +4 -0
  29. data/spec/dummy/config/application.rb +18 -0
  30. data/spec/dummy/config/boot.rb +5 -0
  31. data/spec/dummy/config/database.yml +25 -0
  32. data/spec/dummy/config/environment.rb +5 -0
  33. data/spec/dummy/config/environments/development.rb +38 -0
  34. data/spec/dummy/config/environments/production.rb +79 -0
  35. data/spec/dummy/config/environments/test.rb +37 -0
  36. data/spec/dummy/config/initializers/assets.rb +11 -0
  37. data/spec/dummy/config/initializers/backtrace_silencers.rb +7 -0
  38. data/spec/dummy/config/initializers/cookies_serializer.rb +3 -0
  39. data/spec/dummy/config/initializers/filter_parameter_logging.rb +4 -0
  40. data/spec/dummy/config/initializers/inflections.rb +16 -0
  41. data/spec/dummy/config/initializers/mime_types.rb +4 -0
  42. data/spec/dummy/config/initializers/session_store.rb +3 -0
  43. data/spec/dummy/config/initializers/wrap_parameters.rb +14 -0
  44. data/spec/dummy/config/locales/en.yml +23 -0
  45. data/spec/dummy/config/routes.rb +3 -0
  46. data/spec/dummy/config/secrets.yml +22 -0
  47. data/spec/dummy/db/development.sqlite3 +0 -0
  48. data/spec/dummy/db/migrate/20150709221900_create_users.rb +11 -0
  49. data/spec/dummy/db/production.sqlite3 +0 -0
  50. data/spec/dummy/db/schema.rb +34 -0
  51. data/spec/dummy/db/test.sqlite3 +0 -0
  52. data/spec/dummy/log/development.log +16 -0
  53. data/spec/dummy/log/test.log +8350 -0
  54. data/spec/dummy/public/404.html +67 -0
  55. data/spec/dummy/public/422.html +67 -0
  56. data/spec/dummy/public/500.html +66 -0
  57. data/spec/dummy/public/favicon.ico +0 -0
  58. data/spec/factories/logins.rb +11 -0
  59. data/spec/models/login_spec.rb +69 -0
  60. data/spec/requests/authenticated_spec.rb +47 -0
  61. data/spec/requests/oauth2_spec.rb +187 -0
  62. data/spec/services/facebook_authenticator_spec.rb +54 -0
  63. data/spec/spec_helper.rb +19 -0
  64. data/spec/support/factory_girl.rb +3 -0
  65. metadata +233 -0
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: b165743edc12371fd7870892ad34aacf10bfe984
4
+ data.tar.gz: ced13aee635701bb2385267b33b952237602217a
5
+ SHA512:
6
+ metadata.gz: 70632344b8bdec1043b8f5899e98166426301af14aaddd5dcea1322454a53e00e8f28a43f4c115ec00ec17fa1a396c8ded1393abcc98a0c6a00e1f05bea36649
7
+ data.tar.gz: 2bc812bf057509fe0df986881be6446fe2fe9430c6268dbfca3d7e7ea2589be3fa695d2a055abb0112558bed86446c3b163f18781070d0783d9f5d52dc755d64
data/LICENSE ADDED
@@ -0,0 +1,22 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2015 simplabs GmbH
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
22
+
data/README.md ADDED
@@ -0,0 +1,3 @@
1
+ = RailsApiAuth
2
+
3
+ This project rocks and uses MIT-LICENSE.
data/Rakefile ADDED
@@ -0,0 +1,20 @@
1
+ begin
2
+ require 'bundler/setup'
3
+ rescue LoadError
4
+ puts 'You must `gem install bundler` and `bundle install` to run rake tasks'
5
+ end
6
+
7
+ APP_RAKEFILE = File.expand_path('../spec/dummy/Rakefile', __FILE__)
8
+ load 'rails/tasks/engine.rake'
9
+
10
+ load 'rails/tasks/statistics.rake'
11
+
12
+ Bundler::GemHelper.install_tasks
13
+
14
+ require 'rspec/core'
15
+ require 'rspec/core/rake_task'
16
+
17
+ desc 'Run all specs in spec directory (excluding plugin specs)'
18
+ RSpec::Core::RakeTask.new(spec: 'app:db:test:prepare')
19
+
20
+ task default: :spec
@@ -0,0 +1,54 @@
1
+ require 'login_not_found'
2
+
3
+ class FacebookApiError < StandardError; end
4
+
5
+ class Oauth2Controller < ApplicationController
6
+
7
+ def create
8
+ case params[:grant_type]
9
+ when 'password'
10
+ authenticate_with_credentials(params[:username], params[:password])
11
+ when 'facebook_auth_code'
12
+ authenticate_with_facebook(params[:auth_code])
13
+ else
14
+ oauth2_error('unsupported_grant_type')
15
+ end
16
+ end
17
+
18
+ def destroy
19
+ oauth2_error('unsupported_token_type') && return unless params[:token_type_hint] == 'access_token'
20
+
21
+ login = Login.find_by(oauth2_token: params[:token]) || LoginNotFound.new
22
+ login.refresh_oauth2_token!
23
+
24
+ head 200
25
+ end
26
+
27
+ private
28
+
29
+ def authenticate_with_credentials(email, password)
30
+ login = Login.find_by(email: email) || LoginNotFound.new
31
+
32
+ if login.authenticate(password)
33
+ render json: { access_token: login.oauth2_token }
34
+ else
35
+ oauth2_error('invalid_grant')
36
+ end
37
+ end
38
+
39
+ def authenticate_with_facebook(auth_code)
40
+ oauth2_error('no_authorization_code') && return unless auth_code.present?
41
+
42
+ login = FacebookAuthenticator.new(auth_code).authenticate
43
+
44
+ render json: { access_token: login.oauth2_token }
45
+
46
+ rescue FacebookApiError
47
+ render nothing: true, status: 500
48
+ end
49
+
50
+ def oauth2_error(error)
51
+ render json: { error: error }, status: 400
52
+ end
53
+
54
+ end
@@ -0,0 +1,7 @@
1
+ module RailsApiAuth
2
+
3
+ class ApplicationController < ActionController::Base
4
+
5
+ end
6
+
7
+ end
@@ -0,0 +1,9 @@
1
+ class LoginNotFound
2
+
3
+ def authenticate(_)
4
+ false
5
+ end
6
+
7
+ def refresh_oauth2_token!; end
8
+
9
+ end
@@ -0,0 +1,47 @@
1
+ require 'email_validator'
2
+
3
+ class Login < ActiveRecord::Base
4
+
5
+ class AlreadyVerifiedError < StandardError; end
6
+ class InvalidSingleUseOAuth2Token < StandardError; end
7
+
8
+ has_secure_password validations: false
9
+
10
+ validates :email, presence: true, email: true
11
+ validates :oauth2_token, presence: true
12
+ validates :single_use_oauth2_token, presence: true
13
+ validates :password, length: { maximum: ActiveModel::SecurePassword::MAX_PASSWORD_LENGTH_ALLOWED }, confirmation: true
14
+ validate :password_or_facebook_uid_present
15
+
16
+ before_validation :ensure_oauth2_token
17
+ before_validation :refresh_single_use_oauth2_token
18
+
19
+ def refresh_oauth2_token!
20
+ ensure_oauth2_token(true)
21
+ save!
22
+ end
23
+
24
+ def consume_single_use_oauth2_token!(token)
25
+ raise InvalidSingleUseOAuth2Token.new if token != single_use_oauth2_token
26
+ refresh_single_use_oauth2_token
27
+ save!
28
+ end
29
+
30
+ private
31
+
32
+ def password_or_facebook_uid_present
33
+ if password_digest.blank? && facebook_uid.blank?
34
+ errors.add :base, 'either password_digest or facebook_uid must be present'
35
+ end
36
+ end
37
+
38
+ def ensure_oauth2_token(force = false)
39
+ set_token = oauth2_token.blank? || force
40
+ self.oauth2_token = SecureRandom.hex(125) if set_token
41
+ end
42
+
43
+ def refresh_single_use_oauth2_token
44
+ self.single_use_oauth2_token = SecureRandom.hex(125)
45
+ end
46
+
47
+ end
@@ -0,0 +1,64 @@
1
+ require 'httparty'
2
+
3
+ class FacebookAuthenticator
4
+
5
+ def initialize(auth_code)
6
+ @auth_code = auth_code
7
+ end
8
+
9
+ def authenticate
10
+ if login.present?
11
+ connect_login_to_fb_account
12
+ else
13
+ create_login_from_fb_account
14
+ end
15
+
16
+ login
17
+ end
18
+
19
+ private
20
+
21
+ def login
22
+ @login ||= Login.find_by(email: facebook_user[:email])
23
+ end
24
+
25
+ def connect_login_to_fb_account
26
+ login.update_attributes!(facebook_uid: facebook_user[:id])
27
+ end
28
+
29
+ def create_login_from_fb_account
30
+ login_attributes = {
31
+ email: facebook_user[:email],
32
+ facebook_uid: facebook_user[:id]
33
+ }
34
+
35
+ @login = Login.create!(login_attributes)
36
+ end
37
+
38
+ def facebook_user
39
+ @facebook_user ||= begin
40
+ access_token = facebook_request(fb_token_url).parsed_response['access_token']
41
+ facebook_request(fb_user_url(access_token)).parsed_response.symbolize_keys
42
+ end
43
+ end
44
+
45
+ def facebook_request(url)
46
+ response = HTTParty.get(url)
47
+ raise FacebookApiError.new if response.code != 200
48
+ response
49
+ end
50
+
51
+ def fb_token_url
52
+ "#{Rails.application.config.x.facebook.graph_url}/oauth/access_token".tap do |url|
53
+ url << "?client_id=#{Rails.application.config.x.facebook.app_id}"
54
+ url << "&redirect_uri=#{Rails.application.config.x.facebook.redirect_uri}"
55
+ url << "&client_secret=#{Rails.application.config.x.facebook.app_secret}"
56
+ url << "&code=#{@auth_code}"
57
+ end
58
+ end
59
+
60
+ def fb_user_url(access_token)
61
+ "#{Rails.application.config.x.facebook.graph_url}/me?access_token=#{access_token}"
62
+ end
63
+
64
+ end
@@ -0,0 +1,6 @@
1
+ Rails.application.config.x.facebook.tap do |facebook|
2
+ facebook.app_id = ENV['FB_APP_ID']
3
+ facebook.app_secret = ENV['FB_APP_SECRET']
4
+ facebook.graph_url = 'https://graph.facebook.com/v2.3'
5
+ facebook.redirect_uri = 'http://localhost:4200/login'
6
+ end
data/config/routes.rb ADDED
@@ -0,0 +1,4 @@
1
+ Rails.application.routes.draw do
2
+ post 'token', to: 'oauth2#create'
3
+ post 'revoke', to: 'oauth2#destroy'
4
+ end
@@ -0,0 +1,16 @@
1
+ class CreateLogins < ActiveRecord::Migration
2
+
3
+ def change
4
+ create_table :logins do |t|
5
+ t.string :email, null: false
6
+ t.string :password_digest, null: true
7
+ t.string :oauth2_token, null: false
8
+ t.string :facebook_uid
9
+ t.string :single_use_oauth2_token
10
+ t.references :user
11
+
12
+ t.timestamps
13
+ end
14
+ end
15
+
16
+ end
@@ -0,0 +1,5 @@
1
+ require 'rails_api_auth/engine'
2
+
3
+ module RailsApiAuth
4
+
5
+ end
@@ -0,0 +1,32 @@
1
+ module RailsApiAuth
2
+
3
+ module Authentication
4
+
5
+ class RequestForbidden < StandardError; end
6
+
7
+ extend ActiveSupport::Concern
8
+
9
+ included do
10
+ attr_reader :current_login
11
+
12
+ rescue_from RequestForbidden, with: :deny_access
13
+
14
+ private
15
+
16
+ def deny_access
17
+ head 403
18
+ end
19
+
20
+ def authenticate!
21
+ auth_header = request.headers[:authorization]
22
+ token = auth_header ? auth_header.split(' ').last : ''
23
+ @current_login ||= Login.find_by!(oauth2_token: token)
24
+
25
+ rescue ActiveRecord::RecordNotFound
26
+ head 401
27
+ end
28
+ end
29
+
30
+ end
31
+
32
+ end
@@ -0,0 +1,19 @@
1
+ module RailsApiAuth
2
+
3
+ class Engine < ::Rails::Engine
4
+
5
+ initializer :append_migrations do |app|
6
+ unless app.root.to_s.match root.to_s
7
+ config.paths['db/migrate'].expanded.each do |expanded_path|
8
+ app.config.paths['db/migrate'] << expanded_path
9
+ end
10
+ end
11
+ end
12
+
13
+ config.generators do |g|
14
+ g.test_framework :rspec
15
+ end
16
+
17
+ end
18
+
19
+ end
@@ -0,0 +1,5 @@
1
+ module RailsApiAuth
2
+
3
+ VERSION = '0.0.2'
4
+
5
+ end
@@ -0,0 +1,4 @@
1
+ # desc "Explaining what the task does"
2
+ # task :rails_api_auth do
3
+ # # Task goes here
4
+ # end
@@ -0,0 +1,28 @@
1
+ == README
2
+
3
+ This README would normally document whatever steps are necessary to get the
4
+ application up and running.
5
+
6
+ Things you may want to cover:
7
+
8
+ * Ruby version
9
+
10
+ * System dependencies
11
+
12
+ * Configuration
13
+
14
+ * Database creation
15
+
16
+ * Database initialization
17
+
18
+ * How to run the test suite
19
+
20
+ * Services (job queues, cache servers, search engines, etc.)
21
+
22
+ * Deployment instructions
23
+
24
+ * ...
25
+
26
+
27
+ Please feel free to use a different markup language if you do not plan to run
28
+ <tt>rake doc:app</tt>.
@@ -0,0 +1,6 @@
1
+ # Add your own tasks in files placed in lib/tasks ending in .rake,
2
+ # for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.
3
+
4
+ require File.expand_path('../config/application', __FILE__)
5
+
6
+ Rails.application.load_tasks
@@ -0,0 +1,13 @@
1
+ // This is a manifest file that'll be compiled into application.js, which will include all the files
2
+ // listed below.
3
+ //
4
+ // Any JavaScript/Coffee file within this directory, lib/assets/javascripts, vendor/assets/javascripts,
5
+ // or any plugin's vendor/assets/javascripts directory can be referenced here using a relative path.
6
+ //
7
+ // It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the
8
+ // compiled file.
9
+ //
10
+ // Read Sprockets README (https://github.com/rails/sprockets#sprockets-directives) for details
11
+ // about supported directives.
12
+ //
13
+ //= require_tree .
@@ -0,0 +1,15 @@
1
+ /*
2
+ * This is a manifest file that'll be compiled into application.css, which will include all the files
3
+ * listed below.
4
+ *
5
+ * Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets,
6
+ * or any plugin's vendor/assets/stylesheets directory can be referenced here using a relative path.
7
+ *
8
+ * You're free to add application-wide styles to this file and they'll appear at the bottom of the
9
+ * compiled file so the styles you add here take precedence over styles defined in any styles
10
+ * defined in the other CSS/SCSS files in this directory. It is generally better to create a new
11
+ * file per style scope.
12
+ *
13
+ *= require_tree .
14
+ *= require_self
15
+ */
@@ -0,0 +1,7 @@
1
+ class ApplicationController < ActionController::Base
2
+
3
+ # Prevent CSRF attacks by raising an exception.
4
+ # For APIs, you may want to use :null_session instead.
5
+ protect_from_forgery with: :exception
6
+
7
+ end
@@ -0,0 +1,13 @@
1
+ require 'rails_api_auth/authentication'
2
+
3
+ class AuthenticatedController < ApplicationController
4
+
5
+ include RailsApiAuth::Authentication
6
+
7
+ before_action :authenticate!
8
+
9
+ def index
10
+ render text: 'zuper content'
11
+ end
12
+
13
+ end