pwned 2.0.1 → 2.3.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/.github/FUNDING.yml +1 -0
- data/.github/workflows/tests.yml +39 -0
- data/CHANGELOG.md +33 -2
- data/README.md +127 -11
- data/bin/pwned +3 -3
- data/lib/pwned/hashed_password.rb +40 -0
- data/lib/pwned/password.rb +9 -123
- data/lib/pwned/password_base.rb +142 -0
- data/lib/pwned/version.rb +1 -1
- data/lib/pwned.rb +24 -2
- data/pwned.gemspec +2 -2
- metadata +13 -31
- data/.travis.yml +0 -27
- data/docs/NotPwnedValidator.html +0 -494
- data/docs/Pwned/Error.html +0 -149
- data/docs/Pwned/Password.html +0 -936
- data/docs/Pwned/TimeoutError.html +0 -152
- data/docs/Pwned.html +0 -521
- data/docs/PwnedValidator.html +0 -192
- data/docs/_index.html +0 -162
- data/docs/class_list.html +0 -51
- data/docs/css/common.css +0 -1
- data/docs/css/full_list.css +0 -58
- data/docs/css/style.css +0 -496
- data/docs/file.README.html +0 -424
- data/docs/file_list.html +0 -56
- data/docs/frames.html +0 -17
- data/docs/index.html +0 -424
- data/docs/js/app.js +0 -303
- data/docs/js/full_list.js +0 -216
- data/docs/js/jquery.js +0 -4
- data/docs/method_list.html +0 -115
- data/docs/top-level-namespace.html +0 -112
@@ -0,0 +1,142 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "digest"
|
4
|
+
require "net/http"
|
5
|
+
|
6
|
+
module Pwned
|
7
|
+
##
|
8
|
+
# This class represents a password. It does all the work of talking to the
|
9
|
+
# Pwned Passwords API to find out if the password has been pwned.
|
10
|
+
# @see https://haveibeenpwned.com/API/v2#PwnedPasswords
|
11
|
+
module PasswordBase
|
12
|
+
##
|
13
|
+
# The base URL for the Pwned Passwords API
|
14
|
+
API_URL = "https://api.pwnedpasswords.com/range/"
|
15
|
+
|
16
|
+
##
|
17
|
+
# The number of characters from the start of the hash of the password that
|
18
|
+
# are used to search for the range of passwords.
|
19
|
+
HASH_PREFIX_LENGTH = 5
|
20
|
+
|
21
|
+
##
|
22
|
+
# The total length of a SHA1 hash
|
23
|
+
SHA1_LENGTH = 40
|
24
|
+
|
25
|
+
##
|
26
|
+
# The default request headers that are used to make HTTP requests to the
|
27
|
+
# API. A user agent is provided as requested in the documentation.
|
28
|
+
# @see https://haveibeenpwned.com/API/v2#UserAgent
|
29
|
+
DEFAULT_REQUEST_HEADERS = {
|
30
|
+
"User-Agent" => "Ruby Pwned::Password #{Pwned::VERSION}"
|
31
|
+
}.freeze
|
32
|
+
|
33
|
+
##
|
34
|
+
# @example
|
35
|
+
# password = Pwned::Password.new("password")
|
36
|
+
# password.pwned? #=> true
|
37
|
+
# password.pwned? #=> true
|
38
|
+
#
|
39
|
+
# @return [Boolean] +true+ when the password has been pwned.
|
40
|
+
# @raise [Pwned::Error] if there are errors with the HTTP request.
|
41
|
+
# @raise [Pwned::TimeoutError] if the HTTP request times out.
|
42
|
+
# @since 1.0.0
|
43
|
+
def pwned?
|
44
|
+
pwned_count > 0
|
45
|
+
end
|
46
|
+
|
47
|
+
##
|
48
|
+
# @example
|
49
|
+
# password = Pwned::Password.new("password")
|
50
|
+
# password.pwned_count #=> 3303003
|
51
|
+
#
|
52
|
+
# @return [Integer] the number of times the password has been pwned.
|
53
|
+
# @raise [Pwned::Error] if there are errors with the HTTP request.
|
54
|
+
# @raise [Pwned::TimeoutError] if the HTTP request times out.
|
55
|
+
# @since 1.0.0
|
56
|
+
def pwned_count
|
57
|
+
@pwned_count ||= fetch_pwned_count
|
58
|
+
end
|
59
|
+
|
60
|
+
##
|
61
|
+
# Returns the full SHA1 hash of the given password in uppercase.
|
62
|
+
# @return [String] The full SHA1 hash of the given password.
|
63
|
+
# @since 1.0.0
|
64
|
+
attr_reader :hashed_password
|
65
|
+
|
66
|
+
private
|
67
|
+
|
68
|
+
attr_reader :request_options, :request_headers, :request_proxy, :ignore_env_proxy
|
69
|
+
|
70
|
+
def fetch_pwned_count
|
71
|
+
for_each_response_line do |line|
|
72
|
+
next unless line.start_with?(hashed_password_suffix)
|
73
|
+
# Count starts after the suffix, followed by a colon
|
74
|
+
return line[(SHA1_LENGTH-HASH_PREFIX_LENGTH+1)..-1].to_i
|
75
|
+
end
|
76
|
+
|
77
|
+
# The hash was not found, we can assume the password is not pwned [yet]
|
78
|
+
0
|
79
|
+
end
|
80
|
+
|
81
|
+
def for_each_response_line(&block)
|
82
|
+
begin
|
83
|
+
with_http_response "#{API_URL}#{hashed_password_prefix}" do |response|
|
84
|
+
response.value # raise if request was unsuccessful
|
85
|
+
stream_response_lines(response, &block)
|
86
|
+
end
|
87
|
+
rescue Timeout::Error => e
|
88
|
+
raise Pwned::TimeoutError, e.message
|
89
|
+
rescue => e
|
90
|
+
raise Pwned::Error, e.message
|
91
|
+
end
|
92
|
+
end
|
93
|
+
|
94
|
+
def hashed_password_prefix
|
95
|
+
@hashed_password[0...HASH_PREFIX_LENGTH]
|
96
|
+
end
|
97
|
+
|
98
|
+
def hashed_password_suffix
|
99
|
+
@hashed_password[HASH_PREFIX_LENGTH..-1]
|
100
|
+
end
|
101
|
+
|
102
|
+
# Make a HTTP GET request given the url and headers.
|
103
|
+
# Yields a `Net::HTTPResponse`.
|
104
|
+
def with_http_response(url, &block)
|
105
|
+
uri = URI(url)
|
106
|
+
|
107
|
+
request = Net::HTTP::Get.new(uri)
|
108
|
+
request.initialize_http_header(request_headers)
|
109
|
+
request_options[:use_ssl] = true
|
110
|
+
|
111
|
+
environment_proxy = ignore_env_proxy ? nil : :ENV
|
112
|
+
|
113
|
+
Net::HTTP.start(
|
114
|
+
uri.host,
|
115
|
+
uri.port,
|
116
|
+
request_proxy&.host || environment_proxy,
|
117
|
+
request_proxy&.port,
|
118
|
+
request_proxy&.user,
|
119
|
+
request_proxy&.password,
|
120
|
+
request_options
|
121
|
+
) do |http|
|
122
|
+
http.request(request, &block)
|
123
|
+
end
|
124
|
+
end
|
125
|
+
|
126
|
+
# Stream a Net::HTTPResponse by line, handling lines that cross chunks.
|
127
|
+
def stream_response_lines(response, &block)
|
128
|
+
last_line = ""
|
129
|
+
|
130
|
+
response.read_body do |chunk|
|
131
|
+
chunk_lines = (last_line + chunk).lines
|
132
|
+
# This could end with half a line, so save it for next time. If
|
133
|
+
# chunk_lines is empty, pop returns nil, so this also ensures last_line
|
134
|
+
# is always a string.
|
135
|
+
last_line = chunk_lines.pop || ""
|
136
|
+
chunk_lines.each(&block)
|
137
|
+
end
|
138
|
+
|
139
|
+
yield last_line unless last_line.empty?
|
140
|
+
end
|
141
|
+
end
|
142
|
+
end
|
data/lib/pwned/version.rb
CHANGED
data/lib/pwned.rb
CHANGED
@@ -1,8 +1,10 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require "digest"
|
3
4
|
require "pwned/version"
|
4
5
|
require "pwned/error"
|
5
6
|
require "pwned/password"
|
7
|
+
require "pwned/hashed_password"
|
6
8
|
|
7
9
|
begin
|
8
10
|
# Load Rails and our custom validator
|
@@ -31,8 +33,11 @@ module Pwned
|
|
31
33
|
# @param password [String] The password you want to check against the API.
|
32
34
|
# @param [Hash] request_options Options that can be passed to +Net::HTTP.start+ when
|
33
35
|
# calling the API
|
34
|
-
# @option request_options [Symbol] :headers ({ "User-Agent" =>
|
36
|
+
# @option request_options [Symbol] :headers ({ "User-Agent" => "Ruby Pwned::Password #{Pwned::VERSION}" })
|
35
37
|
# HTTP headers to include in the request
|
38
|
+
# @option request_options [Symbol] :ignore_env_proxy (false) The library
|
39
|
+
# will try to infer an HTTP proxy from the `http_proxy` environment
|
40
|
+
# variable. If you do not want this behaviour, set this option to true.
|
36
41
|
# @return [Boolean] Whether the password appears in the data breaches or not.
|
37
42
|
# @since 1.1.0
|
38
43
|
def self.pwned?(password, request_options={})
|
@@ -49,12 +54,29 @@ module Pwned
|
|
49
54
|
# @param password [String] The password you want to check against the API.
|
50
55
|
# @param [Hash] request_options Options that can be passed to +Net::HTTP.start+ when
|
51
56
|
# calling the API
|
52
|
-
# @option request_options [Symbol] :headers ({ "User-Agent" =>
|
57
|
+
# @option request_options [Symbol] :headers ({ "User-Agent" => "Ruby Pwned::Password #{Pwned::VERSION}" })
|
53
58
|
# HTTP headers to include in the request
|
59
|
+
# @option request_options [Symbol] :ignore_env_proxy (false) The library
|
60
|
+
# will try to infer an HTTP proxy from the `http_proxy` environment
|
61
|
+
# variable. If you do not want this behaviour, set this option to true.
|
54
62
|
# @return [Integer] The number of times the password has appeared in the data
|
55
63
|
# breaches.
|
56
64
|
# @since 1.1.0
|
57
65
|
def self.pwned_count(password, request_options={})
|
58
66
|
Pwned::Password.new(password, request_options).pwned_count
|
59
67
|
end
|
68
|
+
|
69
|
+
##
|
70
|
+
# Returns the full SHA1 hash of the given password in uppercase. This can be safely passed around your code
|
71
|
+
# before making the pwned request (e.g. dropped into a queue table).
|
72
|
+
#
|
73
|
+
# @example
|
74
|
+
# Pwned.hash_password("password") #=> 5BAA61E4C9B93F3F0682250B6CF8331B7EE68FD8
|
75
|
+
#
|
76
|
+
# @param password [String] The password you want to check against the API
|
77
|
+
# @return [String] An uppercase SHA1 hash of the password
|
78
|
+
# @since 2.1.0
|
79
|
+
def self.hash_password(password)
|
80
|
+
Digest::SHA1.hexdigest(password).upcase
|
81
|
+
end
|
60
82
|
end
|
data/pwned.gemspec
CHANGED
@@ -16,7 +16,7 @@ Gem::Specification.new do |spec|
|
|
16
16
|
spec.metadata = {
|
17
17
|
"bug_tracker_uri" => "https://github.com/philnash/pwned/issues",
|
18
18
|
"change_log_uri" => "https://github.com/philnash/pwned/blob/master/CHANGELOG.md",
|
19
|
-
"documentation_uri" => "https://
|
19
|
+
"documentation_uri" => "https://www.rubydoc.info/gems/pwned",
|
20
20
|
"homepage_uri" => "https://github.com/philnash/pwned",
|
21
21
|
"source_code_uri" => "https://github.com/philnash/pwned"
|
22
22
|
}
|
@@ -28,7 +28,7 @@ Gem::Specification.new do |spec|
|
|
28
28
|
spec.executables = ["pwned"]
|
29
29
|
|
30
30
|
spec.add_development_dependency "bundler", ">= 1.16", "< 3.0"
|
31
|
-
spec.add_development_dependency "rake", "~>
|
31
|
+
spec.add_development_dependency "rake", "~> 13.0"
|
32
32
|
spec.add_development_dependency "rspec", "~> 3.0"
|
33
33
|
spec.add_development_dependency "webmock", "~> 3.3"
|
34
34
|
spec.add_development_dependency "yard", "~> 0.9.12"
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: pwned
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.0
|
4
|
+
version: 2.3.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Phil Nash
|
8
|
-
autorequire:
|
8
|
+
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2021-08-30 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: bundler
|
@@ -36,14 +36,14 @@ dependencies:
|
|
36
36
|
requirements:
|
37
37
|
- - "~>"
|
38
38
|
- !ruby/object:Gem::Version
|
39
|
-
version: '
|
39
|
+
version: '13.0'
|
40
40
|
type: :development
|
41
41
|
prerelease: false
|
42
42
|
version_requirements: !ruby/object:Gem::Requirement
|
43
43
|
requirements:
|
44
44
|
- - "~>"
|
45
45
|
- !ruby/object:Gem::Version
|
46
|
-
version: '
|
46
|
+
version: '13.0'
|
47
47
|
- !ruby/object:Gem::Dependency
|
48
48
|
name: rspec
|
49
49
|
requirement: !ruby/object:Gem::Requirement
|
@@ -94,9 +94,10 @@ executables:
|
|
94
94
|
extensions: []
|
95
95
|
extra_rdoc_files: []
|
96
96
|
files:
|
97
|
+
- ".github/FUNDING.yml"
|
98
|
+
- ".github/workflows/tests.yml"
|
97
99
|
- ".gitignore"
|
98
100
|
- ".rspec"
|
99
|
-
- ".travis.yml"
|
100
101
|
- ".yardopts"
|
101
102
|
- CHANGELOG.md
|
102
103
|
- CODE_OF_CONDUCT.md
|
@@ -107,31 +108,13 @@ files:
|
|
107
108
|
- bin/console
|
108
109
|
- bin/pwned
|
109
110
|
- bin/setup
|
110
|
-
- docs/NotPwnedValidator.html
|
111
|
-
- docs/Pwned.html
|
112
|
-
- docs/Pwned/Error.html
|
113
|
-
- docs/Pwned/Password.html
|
114
|
-
- docs/Pwned/TimeoutError.html
|
115
|
-
- docs/PwnedValidator.html
|
116
|
-
- docs/_index.html
|
117
|
-
- docs/class_list.html
|
118
|
-
- docs/css/common.css
|
119
|
-
- docs/css/full_list.css
|
120
|
-
- docs/css/style.css
|
121
|
-
- docs/file.README.html
|
122
|
-
- docs/file_list.html
|
123
|
-
- docs/frames.html
|
124
|
-
- docs/index.html
|
125
|
-
- docs/js/app.js
|
126
|
-
- docs/js/full_list.js
|
127
|
-
- docs/js/jquery.js
|
128
|
-
- docs/method_list.html
|
129
|
-
- docs/top-level-namespace.html
|
130
111
|
- lib/locale/en.yml
|
131
112
|
- lib/pwned.rb
|
132
113
|
- lib/pwned/error.rb
|
114
|
+
- lib/pwned/hashed_password.rb
|
133
115
|
- lib/pwned/not_pwned_validator.rb
|
134
116
|
- lib/pwned/password.rb
|
117
|
+
- lib/pwned/password_base.rb
|
135
118
|
- lib/pwned/version.rb
|
136
119
|
- pwned.gemspec
|
137
120
|
homepage: https://github.com/philnash/pwned
|
@@ -140,10 +123,10 @@ licenses:
|
|
140
123
|
metadata:
|
141
124
|
bug_tracker_uri: https://github.com/philnash/pwned/issues
|
142
125
|
change_log_uri: https://github.com/philnash/pwned/blob/master/CHANGELOG.md
|
143
|
-
documentation_uri: https://
|
126
|
+
documentation_uri: https://www.rubydoc.info/gems/pwned
|
144
127
|
homepage_uri: https://github.com/philnash/pwned
|
145
128
|
source_code_uri: https://github.com/philnash/pwned
|
146
|
-
post_install_message:
|
129
|
+
post_install_message:
|
147
130
|
rdoc_options: []
|
148
131
|
require_paths:
|
149
132
|
- lib
|
@@ -158,9 +141,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
158
141
|
- !ruby/object:Gem::Version
|
159
142
|
version: '0'
|
160
143
|
requirements: []
|
161
|
-
|
162
|
-
|
163
|
-
signing_key:
|
144
|
+
rubygems_version: 3.1.2
|
145
|
+
signing_key:
|
164
146
|
specification_version: 4
|
165
147
|
summary: Tools to use the Pwned Passwords API.
|
166
148
|
test_files: []
|
data/.travis.yml
DELETED
@@ -1,27 +0,0 @@
|
|
1
|
-
sudo: false
|
2
|
-
language: ruby
|
3
|
-
|
4
|
-
env:
|
5
|
-
matrix:
|
6
|
-
- RAILS_VERSION=4.2.11.1
|
7
|
-
- RAILS_VERSION=5.0.7.2
|
8
|
-
- RAILS_VERSION=5.1.7
|
9
|
-
- RAILS_VERSION=5.2.3
|
10
|
-
- RAILS_VERSION=6.0.0
|
11
|
-
|
12
|
-
rvm:
|
13
|
-
- 2.7
|
14
|
-
- 2.6
|
15
|
-
- 2.5
|
16
|
-
- 2.4
|
17
|
-
- jruby
|
18
|
-
- ruby-head
|
19
|
-
|
20
|
-
before_install: gem install bundler
|
21
|
-
|
22
|
-
matrix:
|
23
|
-
allow_failures:
|
24
|
-
- rvm: ruby-head
|
25
|
-
exclude:
|
26
|
-
- rvm: 2.4
|
27
|
-
env: RAILS_VERSION=6.0.0
|
data/docs/NotPwnedValidator.html
DELETED
@@ -1,494 +0,0 @@
|
|
1
|
-
<!DOCTYPE html>
|
2
|
-
<html>
|
3
|
-
<head>
|
4
|
-
<meta charset="utf-8">
|
5
|
-
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
6
|
-
<title>
|
7
|
-
Class: NotPwnedValidator
|
8
|
-
|
9
|
-
— Documentation by YARD 0.9.20
|
10
|
-
|
11
|
-
</title>
|
12
|
-
|
13
|
-
<link rel="stylesheet" href="css/style.css" type="text/css" charset="utf-8" />
|
14
|
-
|
15
|
-
<link rel="stylesheet" href="css/common.css" type="text/css" charset="utf-8" />
|
16
|
-
|
17
|
-
<script type="text/javascript" charset="utf-8">
|
18
|
-
pathId = "NotPwnedValidator";
|
19
|
-
relpath = '';
|
20
|
-
</script>
|
21
|
-
|
22
|
-
|
23
|
-
<script type="text/javascript" charset="utf-8" src="js/jquery.js"></script>
|
24
|
-
|
25
|
-
<script type="text/javascript" charset="utf-8" src="js/app.js"></script>
|
26
|
-
|
27
|
-
|
28
|
-
</head>
|
29
|
-
<body>
|
30
|
-
<div class="nav_wrap">
|
31
|
-
<iframe id="nav" src="class_list.html?1"></iframe>
|
32
|
-
<div id="resizer"></div>
|
33
|
-
</div>
|
34
|
-
|
35
|
-
<div id="main" tabindex="-1">
|
36
|
-
<div id="header">
|
37
|
-
<div id="menu">
|
38
|
-
|
39
|
-
<a href="_index.html">Index (N)</a> »
|
40
|
-
|
41
|
-
|
42
|
-
<span class="title">NotPwnedValidator</span>
|
43
|
-
|
44
|
-
</div>
|
45
|
-
|
46
|
-
<div id="search">
|
47
|
-
|
48
|
-
<a class="full_list_link" id="class_list_link"
|
49
|
-
href="class_list.html">
|
50
|
-
|
51
|
-
<svg width="24" height="24">
|
52
|
-
<rect x="0" y="4" width="24" height="4" rx="1" ry="1"></rect>
|
53
|
-
<rect x="0" y="12" width="24" height="4" rx="1" ry="1"></rect>
|
54
|
-
<rect x="0" y="20" width="24" height="4" rx="1" ry="1"></rect>
|
55
|
-
</svg>
|
56
|
-
</a>
|
57
|
-
|
58
|
-
</div>
|
59
|
-
<div class="clear"></div>
|
60
|
-
</div>
|
61
|
-
|
62
|
-
<div id="content"><h1>Class: NotPwnedValidator
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
</h1>
|
67
|
-
<div class="box_info">
|
68
|
-
|
69
|
-
<dl>
|
70
|
-
<dt>Inherits:</dt>
|
71
|
-
<dd>
|
72
|
-
<span class="inheritName">ActiveModel::EachValidator</span>
|
73
|
-
|
74
|
-
<ul class="fullTree">
|
75
|
-
<li>Object</li>
|
76
|
-
|
77
|
-
<li class="next">ActiveModel::EachValidator</li>
|
78
|
-
|
79
|
-
<li class="next">NotPwnedValidator</li>
|
80
|
-
|
81
|
-
</ul>
|
82
|
-
<a href="#" class="inheritanceTree">show all</a>
|
83
|
-
|
84
|
-
</dd>
|
85
|
-
</dl>
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
<dl>
|
98
|
-
<dt>Defined in:</dt>
|
99
|
-
<dd>lib/pwned/not_pwned_validator.rb</dd>
|
100
|
-
</dl>
|
101
|
-
|
102
|
-
</div>
|
103
|
-
|
104
|
-
<h2>Overview</h2><div class="docstring">
|
105
|
-
<div class="discussion">
|
106
|
-
|
107
|
-
<p>An <code>ActiveModel</code> validator to check passwords against the Pwned
|
108
|
-
Passwords API.</p>
|
109
|
-
|
110
|
-
|
111
|
-
</div>
|
112
|
-
</div>
|
113
|
-
<div class="tags">
|
114
|
-
|
115
|
-
<div class="examples">
|
116
|
-
<p class="tag_title">Examples:</p>
|
117
|
-
|
118
|
-
|
119
|
-
<p class="example_title"><div class='inline'>
|
120
|
-
<p>Validate a password on a <code>User</code> model with the default options.</p>
|
121
|
-
</div></p>
|
122
|
-
|
123
|
-
<pre class="example code"><code><span class='kw'>class</span> <span class='const'>User</span> <span class='op'><</span> <span class='const'>ApplicationRecord</span>
|
124
|
-
<span class='id identifier rubyid_validates'>validates</span> <span class='symbol'>:password</span><span class='comma'>,</span> <span class='label'>not_pwned:</span> <span class='kw'>true</span>
|
125
|
-
<span class='kw'>end</span></code></pre>
|
126
|
-
|
127
|
-
|
128
|
-
<p class="example_title"><div class='inline'>
|
129
|
-
<p>Validate a password on a <code>User</code> model with a custom error
|
130
|
-
message.</p>
|
131
|
-
</div></p>
|
132
|
-
|
133
|
-
<pre class="example code"><code><span class='kw'>class</span> <span class='const'>User</span> <span class='op'><</span> <span class='const'>ApplicationRecord</span>
|
134
|
-
<span class='id identifier rubyid_validates'>validates</span> <span class='symbol'>:password</span><span class='comma'>,</span> <span class='label'>not_pwned:</span> <span class='lbrace'>{</span> <span class='label'>message:</span> <span class='tstring'><span class='tstring_beg'>"</span><span class='tstring_content'>has been pwned %{count} times</span><span class='tstring_end'>"</span></span> <span class='rbrace'>}</span>
|
135
|
-
<span class='kw'>end</span></code></pre>
|
136
|
-
|
137
|
-
|
138
|
-
<p class="example_title"><div class='inline'>
|
139
|
-
<p>Validate a password on a <code>User</code> model that allows the password
|
140
|
-
to have been breached once.</p>
|
141
|
-
</div></p>
|
142
|
-
|
143
|
-
<pre class="example code"><code><span class='kw'>class</span> <span class='const'>User</span> <span class='op'><</span> <span class='const'>ApplicationRecord</span>
|
144
|
-
<span class='id identifier rubyid_validates'>validates</span> <span class='symbol'>:password</span><span class='comma'>,</span> <span class='label'>not_pwned:</span> <span class='lbrace'>{</span> <span class='label'>threshold:</span> <span class='int'>1</span> <span class='rbrace'>}</span>
|
145
|
-
<span class='kw'>end</span></code></pre>
|
146
|
-
|
147
|
-
|
148
|
-
<p class="example_title"><div class='inline'>
|
149
|
-
<p>Validate a password on a <code>User</code> model, handling API errors in
|
150
|
-
various ways</p>
|
151
|
-
</div></p>
|
152
|
-
|
153
|
-
<pre class="example code"><code><span class='kw'>class</span> <span class='const'>User</span> <span class='op'><</span> <span class='const'>ApplicationRecord</span>
|
154
|
-
<span class='comment'># The record is marked as invalid on network errors
|
155
|
-
</span> <span class='comment'># (error message "could not be verified against the past data breaches".)
|
156
|
-
</span> <span class='id identifier rubyid_validates'>validates</span> <span class='symbol'>:password</span><span class='comma'>,</span> <span class='label'>not_pwned:</span> <span class='lbrace'>{</span> <span class='label'>on_error:</span> <span class='symbol'>:invalid</span> <span class='rbrace'>}</span>
|
157
|
-
|
158
|
-
<span class='comment'># The record is marked as invalid on network errors with custom error.
|
159
|
-
</span> <span class='id identifier rubyid_validates'>validates</span> <span class='symbol'>:password</span><span class='comma'>,</span> <span class='label'>not_pwned:</span> <span class='lbrace'>{</span> <span class='label'>on_error:</span> <span class='symbol'>:invalid</span><span class='comma'>,</span> <span class='label'>error_message:</span> <span class='tstring'><span class='tstring_beg'>"</span><span class='tstring_content'>might be pwned</span><span class='tstring_end'>"</span></span> <span class='rbrace'>}</span>
|
160
|
-
|
161
|
-
<span class='comment'># An error is raised on network errors.
|
162
|
-
</span> <span class='comment'># This means that `record.valid?` will raise `Pwned::Error`.
|
163
|
-
</span> <span class='comment'># Not recommended to use in production.
|
164
|
-
</span> <span class='id identifier rubyid_validates'>validates</span> <span class='symbol'>:password</span><span class='comma'>,</span> <span class='label'>not_pwned:</span> <span class='lbrace'>{</span> <span class='label'>on_error:</span> <span class='symbol'>:raise_error</span> <span class='rbrace'>}</span>
|
165
|
-
|
166
|
-
<span class='comment'># Call custom proc on error. For example, capture errors in Sentry,
|
167
|
-
</span> <span class='comment'># but do not mark the record as invalid.
|
168
|
-
</span> <span class='id identifier rubyid_validates'>validates</span> <span class='symbol'>:password</span><span class='comma'>,</span> <span class='label'>not_pwned:</span> <span class='lbrace'>{</span>
|
169
|
-
<span class='label'>on_error:</span> <span class='tlambda'>-></span><span class='lparen'>(</span><span class='id identifier rubyid_record'>record</span><span class='comma'>,</span> <span class='id identifier rubyid_error'>error</span><span class='rparen'>)</span> <span class='tlambeg'>{</span> <span class='const'>Raven</span><span class='period'>.</span><span class='id identifier rubyid_capture_exception'>capture_exception</span><span class='lparen'>(</span><span class='id identifier rubyid_error'>error</span><span class='rparen'>)</span> <span class='rbrace'>}</span>
|
170
|
-
<span class='rbrace'>}</span>
|
171
|
-
<span class='kw'>end</span></code></pre>
|
172
|
-
|
173
|
-
</div>
|
174
|
-
|
175
|
-
<p class="tag_title">Since:</p>
|
176
|
-
<ul class="since">
|
177
|
-
|
178
|
-
<li>
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
<div class='inline'>
|
185
|
-
<p>1.2.0</p>
|
186
|
-
</div>
|
187
|
-
|
188
|
-
</li>
|
189
|
-
|
190
|
-
</ul>
|
191
|
-
|
192
|
-
</div><div id="subclasses">
|
193
|
-
<h2>Direct Known Subclasses</h2>
|
194
|
-
<p class="children"><span class='object_link'><a href="PwnedValidator.html" title="PwnedValidator (class)">PwnedValidator</a></span></p>
|
195
|
-
</div>
|
196
|
-
|
197
|
-
|
198
|
-
<h2>
|
199
|
-
Constant Summary
|
200
|
-
<small><a href="#" class="constants_summary_toggle">collapse</a></small>
|
201
|
-
</h2>
|
202
|
-
|
203
|
-
<dl class="constants">
|
204
|
-
|
205
|
-
<dt id="DEFAULT_ON_ERROR-constant" class="">DEFAULT_ON_ERROR =
|
206
|
-
<div class="docstring">
|
207
|
-
<div class="discussion">
|
208
|
-
|
209
|
-
<p>The default behaviour of this validator in the case of an API failure. The
|
210
|
-
default will mean that if the API fails the object will not be marked
|
211
|
-
invalid.</p>
|
212
|
-
|
213
|
-
|
214
|
-
</div>
|
215
|
-
</div>
|
216
|
-
<div class="tags">
|
217
|
-
|
218
|
-
<p class="tag_title">Since:</p>
|
219
|
-
<ul class="since">
|
220
|
-
|
221
|
-
<li>
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
<div class='inline'>
|
228
|
-
<p>1.2.0</p>
|
229
|
-
</div>
|
230
|
-
|
231
|
-
</li>
|
232
|
-
|
233
|
-
</ul>
|
234
|
-
|
235
|
-
</div>
|
236
|
-
</dt>
|
237
|
-
<dd><pre class="code"><span class='symbol'>:valid</span></pre></dd>
|
238
|
-
|
239
|
-
<dt id="DEFAULT_THRESHOLD-constant" class="">DEFAULT_THRESHOLD =
|
240
|
-
<div class="docstring">
|
241
|
-
<div class="discussion">
|
242
|
-
|
243
|
-
<p>The default threshold for whether a breach is considered pwned. The default
|
244
|
-
is 0, so any password that appears in a breach will mark the record as
|
245
|
-
invalid.</p>
|
246
|
-
|
247
|
-
|
248
|
-
</div>
|
249
|
-
</div>
|
250
|
-
<div class="tags">
|
251
|
-
|
252
|
-
<p class="tag_title">Since:</p>
|
253
|
-
<ul class="since">
|
254
|
-
|
255
|
-
<li>
|
256
|
-
|
257
|
-
|
258
|
-
|
259
|
-
|
260
|
-
|
261
|
-
<div class='inline'>
|
262
|
-
<p>1.2.0</p>
|
263
|
-
</div>
|
264
|
-
|
265
|
-
</li>
|
266
|
-
|
267
|
-
</ul>
|
268
|
-
|
269
|
-
</div>
|
270
|
-
</dt>
|
271
|
-
<dd><pre class="code"><span class='int'>0</span></pre></dd>
|
272
|
-
|
273
|
-
</dl>
|
274
|
-
|
275
|
-
|
276
|
-
|
277
|
-
|
278
|
-
|
279
|
-
|
280
|
-
|
281
|
-
|
282
|
-
|
283
|
-
<h2>
|
284
|
-
Instance Method Summary
|
285
|
-
<small><a href="#" class="summary_toggle">collapse</a></small>
|
286
|
-
</h2>
|
287
|
-
|
288
|
-
<ul class="summary">
|
289
|
-
|
290
|
-
<li class="public ">
|
291
|
-
<span class="summary_signature">
|
292
|
-
|
293
|
-
<a href="#validate_each-instance_method" title="#validate_each (instance method)">#<strong>validate_each</strong>(record, attribute, value) ⇒ Object </a>
|
294
|
-
|
295
|
-
|
296
|
-
|
297
|
-
</span>
|
298
|
-
|
299
|
-
|
300
|
-
|
301
|
-
|
302
|
-
|
303
|
-
|
304
|
-
|
305
|
-
|
306
|
-
|
307
|
-
<span class="summary_desc"><div class='inline'>
|
308
|
-
<p>Validates the <code>value</code> against the Pwned Passwords API.</p>
|
309
|
-
</div></span>
|
310
|
-
|
311
|
-
</li>
|
312
|
-
|
313
|
-
|
314
|
-
</ul>
|
315
|
-
|
316
|
-
|
317
|
-
|
318
|
-
|
319
|
-
|
320
|
-
<div id="instance_method_details" class="method_details_list">
|
321
|
-
<h2>Instance Method Details</h2>
|
322
|
-
|
323
|
-
|
324
|
-
<div class="method_details first">
|
325
|
-
<h3 class="signature first" id="validate_each-instance_method">
|
326
|
-
|
327
|
-
#<strong>validate_each</strong>(record, attribute, value) ⇒ <tt>Object</tt>
|
328
|
-
|
329
|
-
|
330
|
-
|
331
|
-
|
332
|
-
|
333
|
-
</h3><div class="docstring">
|
334
|
-
<div class="discussion">
|
335
|
-
|
336
|
-
<p>Validates the <code>value</code> against the Pwned Passwords API. If the
|
337
|
-
<code>pwned_count</code> is higher than the optional <code>threshold</code>
|
338
|
-
then the record is marked as invalid.</p>
|
339
|
-
|
340
|
-
<p>In the case of an API error the validator will either mark the record as
|
341
|
-
valid or invalid. Alternatively it will run an associated proc or re-raise
|
342
|
-
the original error.</p>
|
343
|
-
|
344
|
-
<p>The validation will short circuit and return with no errors added if the
|
345
|
-
password is blank. The <code>Pwned::Password</code> initializer expects the
|
346
|
-
password to be a string and will throw a <code>TypeError</code> if it is
|
347
|
-
<code>nil</code>. Also, technically the empty string is not a password that
|
348
|
-
is reported to be found in data breaches, so returns <code>false</code>,
|
349
|
-
short circuiting that using <code>value.blank?</code> saves us a trip to
|
350
|
-
the API.</p>
|
351
|
-
|
352
|
-
|
353
|
-
</div>
|
354
|
-
</div>
|
355
|
-
<div class="tags">
|
356
|
-
<p class="tag_title">Parameters:</p>
|
357
|
-
<ul class="param">
|
358
|
-
|
359
|
-
<li>
|
360
|
-
|
361
|
-
<span class='name'>record</span>
|
362
|
-
|
363
|
-
|
364
|
-
<span class='type'>(<tt>ActiveModel::Validations</tt>)</span>
|
365
|
-
|
366
|
-
|
367
|
-
|
368
|
-
—
|
369
|
-
<div class='inline'>
|
370
|
-
<p>The object being validated</p>
|
371
|
-
</div>
|
372
|
-
|
373
|
-
</li>
|
374
|
-
|
375
|
-
<li>
|
376
|
-
|
377
|
-
<span class='name'>attribute</span>
|
378
|
-
|
379
|
-
|
380
|
-
<span class='type'>(<tt>Symbol</tt>)</span>
|
381
|
-
|
382
|
-
|
383
|
-
|
384
|
-
—
|
385
|
-
<div class='inline'>
|
386
|
-
<p>The attribute on the record that is currently being validated.</p>
|
387
|
-
</div>
|
388
|
-
|
389
|
-
</li>
|
390
|
-
|
391
|
-
<li>
|
392
|
-
|
393
|
-
<span class='name'>value</span>
|
394
|
-
|
395
|
-
|
396
|
-
<span class='type'>(<tt>String</tt>)</span>
|
397
|
-
|
398
|
-
|
399
|
-
|
400
|
-
—
|
401
|
-
<div class='inline'>
|
402
|
-
<p>The value of the attribute on the record that is the subject of the
|
403
|
-
validation</p>
|
404
|
-
</div>
|
405
|
-
|
406
|
-
</li>
|
407
|
-
|
408
|
-
</ul>
|
409
|
-
|
410
|
-
<p class="tag_title">Since:</p>
|
411
|
-
<ul class="since">
|
412
|
-
|
413
|
-
<li>
|
414
|
-
|
415
|
-
|
416
|
-
|
417
|
-
|
418
|
-
|
419
|
-
<div class='inline'>
|
420
|
-
<p>1.2.0</p>
|
421
|
-
</div>
|
422
|
-
|
423
|
-
</li>
|
424
|
-
|
425
|
-
</ul>
|
426
|
-
|
427
|
-
</div><table class="source_code">
|
428
|
-
<tr>
|
429
|
-
<td>
|
430
|
-
<pre class="lines">
|
431
|
-
|
432
|
-
|
433
|
-
77
|
434
|
-
78
|
435
|
-
79
|
436
|
-
80
|
437
|
-
81
|
438
|
-
82
|
439
|
-
83
|
440
|
-
84
|
441
|
-
85
|
442
|
-
86
|
443
|
-
87
|
444
|
-
88
|
445
|
-
89
|
446
|
-
90
|
447
|
-
91
|
448
|
-
92
|
449
|
-
93
|
450
|
-
94
|
451
|
-
95
|
452
|
-
96</pre>
|
453
|
-
</td>
|
454
|
-
<td>
|
455
|
-
<pre class="code"><span class="info file"># File 'lib/pwned/not_pwned_validator.rb', line 77</span>
|
456
|
-
|
457
|
-
<span class='kw'>def</span> <span class='id identifier rubyid_validate_each'>validate_each</span><span class='lparen'>(</span><span class='id identifier rubyid_record'>record</span><span class='comma'>,</span> <span class='id identifier rubyid_attribute'>attribute</span><span class='comma'>,</span> <span class='id identifier rubyid_value'>value</span><span class='rparen'>)</span>
|
458
|
-
<span class='kw'>return</span> <span class='kw'>if</span> <span class='id identifier rubyid_value'>value</span><span class='period'>.</span><span class='id identifier rubyid_blank?'>blank?</span>
|
459
|
-
<span class='kw'>begin</span>
|
460
|
-
<span class='id identifier rubyid_pwned_check'>pwned_check</span> <span class='op'>=</span> <span class='const'><span class='object_link'><a href="Pwned.html" title="Pwned (module)">Pwned</a></span></span><span class='op'>::</span><span class='const'><span class='object_link'><a href="Pwned/Password.html" title="Pwned::Password (class)">Password</a></span></span><span class='period'>.</span><span class='id identifier rubyid_new'><span class='object_link'><a href="Pwned/Password.html#initialize-instance_method" title="Pwned::Password#initialize (method)">new</a></span></span><span class='lparen'>(</span><span class='id identifier rubyid_value'>value</span><span class='comma'>,</span> <span class='id identifier rubyid_request_options'>request_options</span><span class='rparen'>)</span>
|
461
|
-
<span class='kw'>if</span> <span class='id identifier rubyid_pwned_check'>pwned_check</span><span class='period'>.</span><span class='id identifier rubyid_pwned_count'>pwned_count</span> <span class='op'>></span> <span class='id identifier rubyid_threshold'>threshold</span>
|
462
|
-
<span class='id identifier rubyid_record'>record</span><span class='period'>.</span><span class='id identifier rubyid_errors'>errors</span><span class='period'>.</span><span class='id identifier rubyid_add'>add</span><span class='lparen'>(</span><span class='id identifier rubyid_attribute'>attribute</span><span class='comma'>,</span> <span class='symbol'>:not_pwned</span><span class='comma'>,</span> <span class='id identifier rubyid_options'>options</span><span class='period'>.</span><span class='id identifier rubyid_merge'>merge</span><span class='lparen'>(</span><span class='label'>count:</span> <span class='id identifier rubyid_pwned_check'>pwned_check</span><span class='period'>.</span><span class='id identifier rubyid_pwned_count'>pwned_count</span><span class='rparen'>)</span><span class='rparen'>)</span>
|
463
|
-
<span class='kw'>end</span>
|
464
|
-
<span class='kw'>rescue</span> <span class='const'><span class='object_link'><a href="Pwned.html" title="Pwned (module)">Pwned</a></span></span><span class='op'>::</span><span class='const'><span class='object_link'><a href="Pwned/Error.html" title="Pwned::Error (class)">Error</a></span></span> <span class='op'>=></span> <span class='id identifier rubyid_error'>error</span>
|
465
|
-
<span class='kw'>case</span> <span class='id identifier rubyid_on_error'>on_error</span>
|
466
|
-
<span class='kw'>when</span> <span class='symbol'>:invalid</span>
|
467
|
-
<span class='id identifier rubyid_record'>record</span><span class='period'>.</span><span class='id identifier rubyid_errors'>errors</span><span class='period'>.</span><span class='id identifier rubyid_add'>add</span><span class='lparen'>(</span><span class='id identifier rubyid_attribute'>attribute</span><span class='comma'>,</span> <span class='symbol'>:pwned_error</span><span class='comma'>,</span> <span class='id identifier rubyid_options'>options</span><span class='period'>.</span><span class='id identifier rubyid_merge'>merge</span><span class='lparen'>(</span><span class='label'>message:</span> <span class='id identifier rubyid_options'>options</span><span class='lbracket'>[</span><span class='symbol'>:error_message</span><span class='rbracket'>]</span><span class='rparen'>)</span><span class='rparen'>)</span>
|
468
|
-
<span class='kw'>when</span> <span class='symbol'>:valid</span>
|
469
|
-
<span class='comment'># Do nothing, consider the record valid
|
470
|
-
</span> <span class='kw'>when</span> <span class='const'>Proc</span>
|
471
|
-
<span class='id identifier rubyid_on_error'>on_error</span><span class='period'>.</span><span class='id identifier rubyid_call'>call</span><span class='lparen'>(</span><span class='id identifier rubyid_record'>record</span><span class='comma'>,</span> <span class='id identifier rubyid_error'>error</span><span class='rparen'>)</span>
|
472
|
-
<span class='kw'>else</span>
|
473
|
-
<span class='id identifier rubyid_raise'>raise</span>
|
474
|
-
<span class='kw'>end</span>
|
475
|
-
<span class='kw'>end</span>
|
476
|
-
<span class='kw'>end</span></pre>
|
477
|
-
</td>
|
478
|
-
</tr>
|
479
|
-
</table>
|
480
|
-
</div>
|
481
|
-
|
482
|
-
</div>
|
483
|
-
|
484
|
-
</div>
|
485
|
-
|
486
|
-
<div id="footer">
|
487
|
-
Generated on Tue Oct 1 21:19:37 2019 by
|
488
|
-
<a href="http://yardoc.org" title="Yay! A Ruby Documentation Tool" target="_parent">yard</a>
|
489
|
-
0.9.20 (ruby-2.5.5).
|
490
|
-
</div>
|
491
|
-
|
492
|
-
</div>
|
493
|
-
</body>
|
494
|
-
</html>
|