signinable 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 7de1e58aefc74e883667a5030469615849840c67
4
+ data.tar.gz: dc5919405a68c6f8e28e43ef7901093b833fee9d
5
+ SHA512:
6
+ metadata.gz: e9f22cfbf7422d93dddc5679ace3dd9e0b1650330529ef94add9e77d12abc69a3711898320aa9e59fa15b1cee4621d7fb382dfea6d5b6878d9fd335a0c68dabe
7
+ data.tar.gz: d0b5549808e353d889d38cc1a7217571d8da4e0dfa7376432297bcd64eca943907a8b40565b437973a7de301d6a496f71efc980dc8a06965c3574448dd69fa4d
data/MIT-LICENSE ADDED
@@ -0,0 +1,20 @@
1
+ Copyright 2013 Ivan Novozhenets
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.rdoc ADDED
@@ -0,0 +1,3 @@
1
+ = Signinable
2
+
3
+ This project rocks and uses MIT-LICENSE.
data/Rakefile ADDED
@@ -0,0 +1,17 @@
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
+ require 'rdoc/task'
8
+
9
+ RDoc::Task.new(:rdoc) do |rdoc|
10
+ rdoc.rdoc_dir = 'rdoc'
11
+ rdoc.title = 'Signinable'
12
+ rdoc.options << '--line-numbers'
13
+ rdoc.rdoc_files.include('README.rdoc')
14
+ rdoc.rdoc_files.include('lib/**/*.rb')
15
+ end
16
+
17
+ Bundler::GemHelper.install_tasks
@@ -0,0 +1,6 @@
1
+ Description:
2
+ Copies signin.rb to app/models/.
3
+ Copies create_signins.rb to db/migrate
4
+
5
+ Examples:
6
+ `rails generate signinable`
@@ -0,0 +1,15 @@
1
+ require 'rails/generators/migration'
2
+
3
+ class SigninableGenerator < Rails::Generators::Base
4
+ include Rails::Generators::Migration
5
+ source_root File.expand_path('../templates', __FILE__)
6
+
7
+ def self.next_migration_number(path)
8
+ Time.now.utc.strftime("%Y%m%d%H%M%S")
9
+ end
10
+
11
+ def create_model_file
12
+ template "signin.rb", "app/models/signin.rb"
13
+ migration_template "create_signins.rb", "db/migrate/create_signins.rb"
14
+ end
15
+ end
@@ -0,0 +1,21 @@
1
+ class CreateSignins < ActiveRecord::Migration
2
+ def self.up
3
+ create_table :signins do |t|
4
+ t.integer :signinable_id, null: false
5
+ t.string :signinable_type, null: false
6
+ t.string :token, null: false
7
+ t.string :referer, default: ""
8
+ t.string :user_agent, default: ""
9
+ t.string :ip, null: false
10
+ t.datetime :expiration_time
11
+ t.timestamps
12
+ end
13
+
14
+ add_index :signins, [:signinable_id, :signinable_type]
15
+ add_index :signins, :token
16
+ end
17
+
18
+ def self.down
19
+ drop_table :signins
20
+ end
21
+ end
@@ -0,0 +1,20 @@
1
+ class Signin < ActiveRecord::Base
2
+ belongs_to :signinable, polymorphic: true
3
+
4
+ validates :token, presence: true
5
+ validates :ip,
6
+ presence: true,
7
+ format: { with: /\A([1-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])(\.([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])){3}\z/ }
8
+
9
+ before_validation on: :create do
10
+ self.token = SecureRandom.urlsafe_base64(rand(50..100))
11
+ end
12
+
13
+ def expire!
14
+ update_attributes(expiration_time: Time.zone.now)
15
+ end
16
+
17
+ def expired?
18
+ expiration_time && expiration_time <= Time.zone.now
19
+ end
20
+ end
@@ -0,0 +1,71 @@
1
+ module Signinable
2
+ module ModelAdditions
3
+ extend ActiveSupport::Concern
4
+
5
+ module ClassMethods
6
+ def signinable(options = {})
7
+ cattr_accessor :signin_expiration
8
+ cattr_accessor :signin_simultaneous
9
+ cattr_accessor :signin_restrictions
10
+ self.signin_expiration = options[:expiration] || 2.hours
11
+ self.signin_simultaneous = options[:simultaneous] || true
12
+ self.signin_restrictions = (options[:restrictions] && options[:restrictions].is_a?(Array)) ? options[:restrictions] : []
13
+
14
+ has_many :signins, as: :signinable, dependent: :destroy
15
+ end
16
+
17
+ def authenticate_with_token(token, ip, user_agent, referer)
18
+ if(signin = Signin.find_by_token(token))
19
+ if self.signin_expiration > 0
20
+ return nil if signin.expired?
21
+ end
22
+
23
+ unless self.signin_simultaneous
24
+ return nil unless signin == signin.signinable.last_signin
25
+ end
26
+
27
+ return nil unless self.check_signin_permission(signin, ip, user_agent, referer)
28
+ signin.update!(expiration_time: (Time.zone.now + self.signin_expiration)) unless self.signin_expiration == 0
29
+ signin.signinable
30
+ end
31
+ end
32
+
33
+ def check_signin_permission(signin, ip, user_agent, referer)
34
+ signin_permitted?(signin, ip, user_agent, referer)
35
+ end
36
+
37
+ private
38
+ def signin_permitted?(signin, ip, user_agent, referer)
39
+ self.signin_restrictions.each do |field|
40
+ if(local_variables.include?(field.to_sym) && signin.respond_to?("#{field}"))
41
+ return false unless signin.send("#{field}") == eval("#{field}")
42
+ end
43
+ end
44
+
45
+ return true
46
+ end
47
+ end
48
+
49
+ def signin(ip, user_agent, referer)
50
+ expiration_time = self.class.signin_expiration == 0 ? nil : (Time.zone.now + self.class.signin_expiration)
51
+ Signin.create!(signinable: self, ip: ip, referer: referer, user_agent: user_agent, expiration_time: expiration_time).token
52
+ end
53
+
54
+ def signout(token, ip, user_agent, referer)
55
+ if(signin = Signin.find_by_token(token))
56
+ return nil unless self.class.check_signin_permission(signin, ip, user_agent, referer)
57
+ signin.expire!
58
+
59
+ return true
60
+ end
61
+
62
+ return nil
63
+ end
64
+
65
+ def last_signin
66
+ signins.last unless signins.empty?
67
+ end
68
+ end
69
+ end
70
+
71
+ ActiveRecord::Base.send(:include, Signinable::ModelAdditions)
@@ -0,0 +1,3 @@
1
+ module Signinable
2
+ VERSION = "1.0.0"
3
+ end
data/lib/signinable.rb ADDED
@@ -0,0 +1,2 @@
1
+ require 'signinable/version'
2
+ require 'signinable/model_additions'
metadata ADDED
@@ -0,0 +1,193 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: signinable
3
+ version: !ruby/object:Gem::Version
4
+ version: 1.0.0
5
+ platform: ruby
6
+ authors:
7
+ - Ivan Novozhenets
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2014-01-09 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: rails
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: 4.0.0
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: 4.0.0
27
+ - !ruby/object:Gem::Dependency
28
+ name: sqlite3
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rspec-rails
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ">="
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ - !ruby/object:Gem::Dependency
56
+ name: capybara
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ">="
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - ">="
67
+ - !ruby/object:Gem::Version
68
+ version: '0'
69
+ - !ruby/object:Gem::Dependency
70
+ name: guard-rspec
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ">="
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: guard-spork
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - ">="
88
+ - !ruby/object:Gem::Version
89
+ version: '0'
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - ">="
95
+ - !ruby/object:Gem::Version
96
+ version: '0'
97
+ - !ruby/object:Gem::Dependency
98
+ name: shoulda
99
+ requirement: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - ">="
102
+ - !ruby/object:Gem::Version
103
+ version: '0'
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - ">="
109
+ - !ruby/object:Gem::Version
110
+ version: '0'
111
+ - !ruby/object:Gem::Dependency
112
+ name: timecop
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - ">="
116
+ - !ruby/object:Gem::Version
117
+ version: '0'
118
+ type: :development
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ requirements:
122
+ - - ">="
123
+ - !ruby/object:Gem::Version
124
+ version: '0'
125
+ - !ruby/object:Gem::Dependency
126
+ name: factory_girl_rails
127
+ requirement: !ruby/object:Gem::Requirement
128
+ requirements:
129
+ - - ">="
130
+ - !ruby/object:Gem::Version
131
+ version: '0'
132
+ type: :development
133
+ prerelease: false
134
+ version_requirements: !ruby/object:Gem::Requirement
135
+ requirements:
136
+ - - ">="
137
+ - !ruby/object:Gem::Version
138
+ version: '0'
139
+ - !ruby/object:Gem::Dependency
140
+ name: database_cleaner
141
+ requirement: !ruby/object:Gem::Requirement
142
+ requirements:
143
+ - - ">="
144
+ - !ruby/object:Gem::Version
145
+ version: '0'
146
+ type: :development
147
+ prerelease: false
148
+ version_requirements: !ruby/object:Gem::Requirement
149
+ requirements:
150
+ - - ">="
151
+ - !ruby/object:Gem::Version
152
+ version: '0'
153
+ description: Allows authentication with tokens
154
+ email:
155
+ - novozhenets@gmail.com
156
+ executables: []
157
+ extensions: []
158
+ extra_rdoc_files: []
159
+ files:
160
+ - MIT-LICENSE
161
+ - README.rdoc
162
+ - Rakefile
163
+ - lib/generators/signinable/USEGA
164
+ - lib/generators/signinable/signinable_generator.rb
165
+ - lib/generators/signinable/templates/create_signins.rb
166
+ - lib/generators/signinable/templates/signin.rb
167
+ - lib/signinable.rb
168
+ - lib/signinable/model_additions.rb
169
+ - lib/signinable/version.rb
170
+ homepage: https://github.com/novozhenets/signinable
171
+ licenses: []
172
+ metadata: {}
173
+ post_install_message:
174
+ rdoc_options: []
175
+ require_paths:
176
+ - lib
177
+ required_ruby_version: !ruby/object:Gem::Requirement
178
+ requirements:
179
+ - - ">="
180
+ - !ruby/object:Gem::Version
181
+ version: '0'
182
+ required_rubygems_version: !ruby/object:Gem::Requirement
183
+ requirements:
184
+ - - ">="
185
+ - !ruby/object:Gem::Version
186
+ version: '0'
187
+ requirements: []
188
+ rubyforge_project:
189
+ rubygems_version: 2.2.0.rc.1
190
+ signing_key:
191
+ specification_version: 4
192
+ summary: Token based signin
193
+ test_files: []