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 +4 -4
- data/README.md +18 -3
- data/lib/sinatra/canvas_auth.erb +89 -0
- data/lib/sinatra/canvas_auth.rb +69 -17
- data/lib/sinatra/canvas_auth/version.rb +1 -1
- metadata +3 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 829a0ea15cee0cbe68d6e29af2ffb52a2f839903
|
4
|
+
data.tar.gz: 651e5961b52eba8b9423c6acecd0b2f34ecaa0fc
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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,
|
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>
|
data/lib/sinatra/canvas_auth.rb
CHANGED
@@ -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
|
-
|
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']}#{
|
56
|
+
"#{request.env['SCRIPT_NAME']}#{settings.token_path}"
|
31
57
|
|
32
|
-
redirect_params = "client_id=#{
|
58
|
+
redirect_params = "client_id=#{settings.client_id}&" \
|
33
59
|
"response_type=code&" \
|
34
|
-
"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 "#{
|
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
|
-
|
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
|
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
|
-
|
81
|
-
|
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
|
87
|
-
|
88
|
-
|
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
|
-
|
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
|
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
|
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-
|
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
|