-
Notifications
You must be signed in to change notification settings - Fork 3
Expand file tree
/
Copy pathauthenticated_system.rb
More file actions
116 lines (105 loc) · 3.82 KB
/
authenticated_system.rb
File metadata and controls
116 lines (105 loc) · 3.82 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
module AuthenticatedSystem
protected
# Returns true or false if the account is logged in.
# Preloads @current_account with the account model if they're logged in.
def logged_in?
current_account != :false
end
# Accesses the current account from the session. Set it to :false if login fails
# so that future calls do not hit the database.
def current_account
@current_account ||= (login_from_session || login_from_basic_auth || login_from_cookie || :false)
end
# Store the given account id in the session.
def current_account=(new_account)
session[:account_id] = (new_account.nil? || new_account.is_a?(Symbol)) ? nil : new_account.id
@current_account = new_account || :false
end
# Check if the account is authorized
#
# Override this method in your controllers if you want to restrict access
# to only a few actions or if you want to check if the account
# has the correct rights.
#
# Example:
#
# # only allow nonbobs
# def authorized?
# current_account.login != "bob"
# end
def authorized?
logged_in?
end
# Filter method to enforce a login requirement.
#
# To require logins for all actions, use this in your controllers:
#
# before_filter :login_required
#
# To require logins for specific actions, use this in your controllers:
#
# before_filter :login_required, :only => [ :edit, :update ]
#
# To skip this in a subclassed controller:
#
# skip_before_filter :login_required
#
def login_required
authorized? || access_denied
end
# Redirect as appropriate when an access request fails.
#
# The default action is to redirect to the login screen.
#
# Override this method in your controllers if you want to have special
# behavior in case the account is not authorized
# to access the requested action. For example, a popup window might
# simply close itself.
def access_denied
respond_to do |format|
format.html do
store_location
redirect_to new_session_path
end
format.any do
request_http_basic_authentication 'Web Password'
end
end
end
# Store the URI of the current request in the session.
#
# We can return to this location by calling #redirect_back_or_default.
def store_location
session[:return_to] = request.request_uri
end
# Redirect to the URI stored by the most recent store_location call or
# to the passed default.
def redirect_back_or_default(default)
redirect_to(session[:return_to] || default)
session[:return_to] = nil
end
# Inclusion hook to make #current_account and #logged_in?
# available as ActionView helper methods.
def self.included(base)
base.send :helper_method, :current_account, :logged_in?
end
# Called from #current_account. First attempt to login by the account id stored in the session.
def login_from_session
self.current_account = Account.find_by_id(session[:account_id]) if session[:account_id]
end
# Called from #current_account. Now, attempt to login by basic authentication information.
def login_from_basic_auth
authenticate_with_http_basic do |username, password|
self.current_account = Account.authenticate(username, password)
end
end
# Called from #current_account. Finaly, attempt to login by an expiring token in the cookie.
def login_from_cookie
account = cookies[:auth_token] && Account.find_by_remember_token(cookies[:auth_token])
if account && account.remember_token?
account.remember_me
cookies[:auth_token] = { :value => account.remember_token, :expires => account.remember_token_expires_at }
self.current_account = account
end
end
end