sinatra-canvas_auth 0.0.3 → 0.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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 18c5767aed8a836503fa4c9880faa99e5e3b21c1
4
- data.tar.gz: 675e5cbbd76fa389ebd9b63580c5a25454be275d
3
+ metadata.gz: 829a0ea15cee0cbe68d6e29af2ffb52a2f839903
4
+ data.tar.gz: 651e5961b52eba8b9423c6acecd0b2f34ecaa0fc
5
5
  SHA512:
6
- metadata.gz: c002fbf02f0f41127c26aa81d2d7d1cf438ea152e55ec5e8d67d0d103b26aaad69e213d9006b95a1c227e15554e8a6493a031c1a7471f6e269608c3ec914f806
7
- data.tar.gz: c1132a29cc37e8a34f0daada820b865629de65f6af4098883852e945737eb2e23dae937c609949b0f34eccbfc8741f3651f3bb747d4e10bb9af3fc71ecaaa15c
6
+ metadata.gz: f92f0c1a68083d4b58e907b0d0d811037619223e29fbfb40c6c684e005303803b1058552a30b0e8437f83359412c45e5222aa464b2b72e2a008318660d737887
7
+ data.tar.gz: 4d4635ca2e3c66edd45d06dc8cfa211f9f8909ad8c9c58781ebb6054bb707c703edc66a2361aa46a170c7225757b5b007c87730dba60981c14f103b1eb439b3d
data/README.md CHANGED
@@ -105,12 +105,19 @@ CanvasAuth requires a baseline configuration to function, as Canvas API settings
105
105
  Default: [/.*/]
106
106
  To only require authentication for certain routes, they may be explicitly specified here with either strings or regular expression. By default, all app routes will require authentication.
107
107
  ```ruby
108
- set :auth_paths, ['/admin', /^\/courses\/(\d)+$]
108
+ set :auth_paths, ['/admin', /^\/courses\/(\d)+$/]
109
109
  ```
110
110
 
111
111
  Alternative syntax:
112
112
  ```ruby
113
- authenticate '/admin', /^\/courses\/(\d)+$
113
+ authenticate '/admin', /^\/courses\/(\d)+$/
114
+ ```
115
+
116
+ * **public_paths** (Array)
117
+ Default: []
118
+ The inverse of auth_paths, routes matching strings or regexps in this array will not require authentication
119
+ ```ruby
120
+ set :public_paths, ['/homepage', /^\/assets\/.+$/]
114
121
  ```
115
122
 
116
123
  * **unauthorized_redirect** (String)
@@ -126,6 +133,13 @@ CanvasAuth requires a baseline configuration to function, as Canvas API settings
126
133
  ```ruby
127
134
  set :logout_redirect, '/goodbye'
128
135
  ```
136
+
137
+ * **failure_redirect** (String)
138
+ Default: "/login-failure"
139
+ If the user declines to grant the app access to their Canvas account, or the API request for a Canvas token raises an unexpected error, the user will be redirected to this path.
140
+ ```ruby
141
+ set :error_path, '/auth-error'
142
+ ```
129
143
   
130
144
 
131
145
  #### Callbacks
@@ -164,9 +178,10 @@ The following are optional hooks called by CanvasAuth which allow you to customi
164
178
  * GET /canvas-auth-logout
165
179
  * POST /canvas-auth-token
166
180
 
167
- * The following routes are also defined by CanvasAuth, but only as placeholders that may (and should) be overridden by your application. They do not include any functionality and serve only as landing pages to prevent 404ing on the default redirects.
181
+ * The following routes are also defined by CanvasAuth, and may be overridden by your application, should you wish to replace the default view/behavior provided:
168
182
  * GET /unauthorized
169
183
  * GET /logged-out
184
+ * GET /login-failure
170
185
 
171
186
  * All routes defined by CanvasAuth are permanently exempt from the requiring authentication, to avoid redirect loops.
172
187
  ## Contributing
@@ -0,0 +1,89 @@
1
+ <% params ||= {} %>
2
+
3
+ <!DOCTYPE html>
4
+ <html>
5
+ <head>
6
+ <style>
7
+ #box-content {
8
+ background-color: #FFF;
9
+ padding: 24px;
10
+ }
11
+
12
+ #modal-box {
13
+ border-radius: 3px;
14
+ box-shadow: 0 1px 4px 1px rgba(0,0,0,0.4);
15
+ box-sizing: border-box;
16
+ margin: 36px auto 0;
17
+ width: 696px;
18
+ }
19
+
20
+ .btn {
21
+ background-color: #CFB87C;
22
+ color: #2D3B45;
23
+ border: 1px solid;
24
+ border-color: #C7CDD1;
25
+ border-radius: 3px;
26
+ padding: 8px 14px;
27
+ font-size: 14px;
28
+ font-size: .875rem;
29
+ line-height: 20px;
30
+ text-align: center;
31
+ cursor: pointer;
32
+ font-weight: 300;
33
+ }
34
+
35
+ .btn:hover {
36
+ background-color: #c8ae69;
37
+ }
38
+
39
+ body {
40
+ background-color: #2e3c46;
41
+ font-family: "Helvetica Neue",Helvetica,Arial,sans-serif;
42
+ font-weight: 300;
43
+ }
44
+
45
+ form {
46
+ margin-top: 30px;
47
+ }
48
+
49
+ header {
50
+ background-color: #34444f;
51
+ padding: 18px;
52
+ }
53
+
54
+ h2 {
55
+ margin-top: 0;
56
+ font-weight: 300;
57
+ }
58
+
59
+ img {
60
+ width: 140px;
61
+ height: 57px;
62
+ }
63
+ </style>
64
+ </head>
65
+
66
+ <body>
67
+ <div id="modal-box">
68
+ <header>
69
+ <img alt="Canvas by Instructure" src="" />
70
+ </header>
71
+
72
+ <div id="box-content">
73
+ <h2>
74
+ <%= header %>
75
+ </h2>
76
+
77
+ <p>
78
+ <%= message %>
79
+ </p>
80
+
81
+ <% if login_url(params['state']) %>
82
+ <form method='GET' action='<%= login_url(params['state']) %>'>
83
+ <input class='btn' type='submit' value='Return to login'/>
84
+ </form>
85
+ <% end %>
86
+ </div>
87
+ </div>
88
+ </body>
89
+ </html>
@@ -1,18 +1,23 @@
1
1
  require 'sinatra'
2
2
  require 'rest-client'
3
+ require 'securerandom'
3
4
 
4
5
  module Sinatra
5
6
  module CanvasAuth
6
7
 
8
+ class StateError < ::StandardError; end
9
+
7
10
  DEFAULT_SETTINGS = {
8
11
  :auth_paths => [/.*/],
9
12
  :canvas_url => nil,
10
13
  :client_id => nil,
11
14
  :client_secret => nil,
15
+ :failure_redirect => '/login-failure',
12
16
  :login_path => '/canvas-auth-login',
13
17
  :token_path => '/canvas-auth-token',
14
18
  :logout_path => '/canvas-auth-logout',
15
19
  :logout_redirect => '/logged-out',
20
+ :public_paths => [],
16
21
  :unauthorized_redirect => '/unauthorized'
17
22
  }.freeze
18
23
 
@@ -24,14 +29,35 @@ module Sinatra
24
29
  def self.registered(app)
25
30
  self.merge_defaults(app)
26
31
 
32
+ app.helpers do
33
+ def login_url(state = nil)
34
+ return false if request.nil?
35
+ path_elements = [request.env['SCRIPT_NAME'], settings.login_path]
36
+ path_elements << state if state
37
+ File.join(path_elements)
38
+ end
39
+
40
+ def render_view(header='', message='')
41
+ render(:erb, :canvas_auth, {
42
+ :views => File.expand_path(File.dirname(__FILE__)),
43
+ :locals => {
44
+ :header => header,
45
+ :message => message
46
+ }
47
+ })
48
+ end
49
+ end
50
+
27
51
  app.get app.login_path do
28
- params['state'] ||= request.env['SCRIPT_NAME']
52
+ session['oauth_redirect'] ||= request.env['SCRIPT_NAME']
53
+ session['oauth_state'] = SecureRandom.urlsafe_base64(24)
54
+
29
55
  redirect_uri = "#{request.scheme}://#{request.host_with_port}" \
30
- "#{request.env['SCRIPT_NAME']}#{app.token_path}"
56
+ "#{request.env['SCRIPT_NAME']}#{settings.token_path}"
31
57
 
32
- redirect_params = "client_id=#{app.client_id}&" \
58
+ redirect_params = "client_id=#{settings.client_id}&" \
33
59
  "response_type=code&" \
34
- "state=#{CGI.escape(params['state'])}&" \
60
+ "state=#{session['oauth_state']}&" \
35
61
  "redirect_uri=#{CGI.escape(redirect_uri)}"
36
62
 
37
63
  ['scope', 'purpose', 'force_login', 'unique_id'].each do |optional_param|
@@ -40,7 +66,7 @@ module Sinatra
40
66
  end
41
67
  end
42
68
 
43
- redirect "#{app.canvas_url}/login/oauth2/auth?#{redirect_params}"
69
+ redirect "#{settings.canvas_url}/login/oauth2/auth?#{redirect_params}"
44
70
  end
45
71
 
46
72
  app.get app.token_path do
@@ -50,14 +76,20 @@ module Sinatra
50
76
  :client_secret => settings.client_secret
51
77
  }
52
78
 
53
- response = RestClient.post("#{settings.canvas_url}/login/oauth2/token", payload)
79
+ begin
80
+ CanvasAuth.verify_oauth_state(session, params)
81
+ response = RestClient.post("#{settings.canvas_url}/login/oauth2/token", payload)
82
+ rescue RestClient::Exception, CanvasAuth::StateError => e
83
+ failure_url = File.join(request.env['SCRIPT_NAME'], settings.failure_redirect)
84
+ redirect (failure_url + "?error=#{params[:error] || CGI.escape(e.to_s)}")
85
+ end
86
+
54
87
  response = JSON.parse(response)
55
88
  session['user_id'] = response['user']['id']
56
89
  session['access_token'] = response['access_token']
57
-
58
90
  oauth_callback(response) if self.respond_to?(:oauth_callback)
59
91
 
60
- redirect params['state']
92
+ redirect session['oauth_redirect']
61
93
  end
62
94
 
63
95
  app.get app.logout_path do
@@ -77,24 +109,31 @@ module Sinatra
77
109
  redirect to(settings.logout_redirect)
78
110
  end
79
111
 
80
- # These two routes exist to prevent 404'ing with default options, but
81
- # ideally they should be overridden by the app, or alternate paths given
82
- app.get '/unauthorized' do
83
- 'Your canvas account unauthorized to view this resource'
112
+ app.get app.logout_redirect do
113
+ render_view('Logged out', 'You have been successfully logged out')
84
114
  end
85
115
 
86
- app.get '/logged-out' do
87
- "You have been logged out <a href='canvas-auth-login'>" \
88
- "Click here</a> to log in again."
116
+ app.get app.unauthorized_redirect do
117
+ render_view('Authentication Failed',
118
+ 'Your canvas account is unauthorized to view this resource')
89
119
  end
90
120
 
121
+ app.get app.failure_redirect do
122
+ message = "Login could not be completed."
123
+ if params[:error] && !params[:error].empty?
124
+ message += " (#{CGI.unescape(params[:error])})"
125
+ end
126
+
127
+ render_view("Authentication Failed", message)
128
+ end
91
129
 
92
130
  # Redirect unauthenticated/unauthorized users before hitting app routes
93
131
  app.before do
94
132
  current_path = "#{request.env['SCRIPT_NAME']}#{request.env['PATH_INFO']}"
95
133
  if CanvasAuth.auth_path?(self.settings, current_path, request.env['SCRIPT_NAME'])
96
134
  if session['user_id'].nil?
97
- redirect "#{request.env['SCRIPT_NAME']}#{settings.login_path}?state=#{current_path}"
135
+ session['oauth_redirect'] = current_path
136
+ redirect "#{request.env['SCRIPT_NAME']}#{settings.login_path}"
98
137
  elsif self.respond_to?(:authorized) && !authorized
99
138
  redirect "#{request.env['SCRIPT_NAME']}#{settings.unauthorized_redirect}"
100
139
  end
@@ -105,9 +144,11 @@ module Sinatra
105
144
  # Should the current path ask for authentication or is it public?
106
145
  def self.auth_path?(app, current_path, script_name = '')
107
146
  exempt_paths = [ app.login_path, app.token_path, app.logout_path,
108
- app.logout_redirect, app.unauthorized_redirect ]
147
+ app.logout_redirect, app.unauthorized_redirect,
148
+ app.failure_redirect ]
109
149
 
110
150
  app.auth_paths.select{ |p| current_path.match(p) }.any? &&
151
+ !app.public_paths.select{ |p| current_path.match(p) }.any? &&
111
152
  !exempt_paths.map{|p| File.join(script_name, p)}.include?(current_path)
112
153
  end
113
154
 
@@ -118,6 +159,17 @@ module Sinatra
118
159
  end
119
160
  end
120
161
  end
162
+
163
+ # Verify state param from Canvas is the same one originally sent. Otherwise,
164
+ # unauthorized requests can be made by intercepting the redirect from Canvas
165
+ # to app token_path and tricking an authorized user into accessing the link.
166
+ # http://homakov.blogspot.com/2012/07/saferweb-most-common-oauth2.html
167
+ def self.verify_oauth_state(params, session)
168
+ saved_state = session['oauth_state']
169
+ if saved_state != params['state'] || (saved_state && saved_state.empty?)
170
+ raise CanvasAuth::StateError, 'Invalid OAuth state token provided'
171
+ end
172
+ end
121
173
  end
122
174
 
123
175
  register CanvasAuth
@@ -1,5 +1,5 @@
1
1
  module Sinatra
2
2
  module CanvasAuth
3
- VERSION = "0.0.3"
3
+ VERSION = "0.1.0"
4
4
  end
5
5
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: sinatra-canvas_auth
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.0.3
4
+ version: 0.1.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Connor Ford
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2016-09-27 00:00:00.000000000 Z
11
+ date: 2016-11-10 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: bundler
@@ -73,6 +73,7 @@ extensions: []
73
73
  extra_rdoc_files: []
74
74
  files:
75
75
  - README.md
76
+ - lib/sinatra/canvas_auth.erb
76
77
  - lib/sinatra/canvas_auth.rb
77
78
  - lib/sinatra/canvas_auth/version.rb
78
79
  homepage: https://github.com/cuonline