2015-04-07 17:25:46 -04:00
package main
import (
"os"
"strings"
"testing"
2021-06-29 20:37:03 +02:00
. "github.com/onsi/gomega"
2015-04-07 17:25:46 -04:00
)
Reload authenticated-emails-file upon update
This change extracts the UserMap class from NewValidator() so that its
LoadAuthenticatedEmailsFile() method can be called concurrently. This method
is called by a goroutine containing a fsnotify.Watcher watching the
authenticated emails file.
Watching isn't forever aborted when the authenticated emails file disappears.
The goroutine will call os.Stat() up to twenty times a second if the file is
persistently missing, but that's the pathological case, not the common one.
The common case is that some editors (including Vim) will perform a
rename-and-replace when updating a file, triggering fsnotify.Rename events,
and the file will temporarily disappear. This watcher goroutine handles that
case.
Also, on some platforms (notably Arch Linux), a remove will be preceded by a
fsnotify.Chmod, causing a race between the upcoming fsnotify.Remove and the
call to UserMap.LoadAuthenticatedEmailsFile(). Hence, we treat fsnotify.Chmod
the same as fsnotify.Remove and fsnotify.Rename. There's no significant
penalty to re-adding a file to the watcher.
Also contains the following small changes from the summary of commits below:
- Minor optimization of email domain search
- Fixed api_test.go on Windows
- Add deferred File.Close() calls where needed
- Log error and return if emails file doesn't parse
These are the original commits from #89 squashed into this one:
0c6f2b6 Refactor validator_test to prepare for more tests
e0c792b Add more test cases to validator_test
a9a9d93 Minor optimization of email domain search
b763ea5 Extract LoadAuthenticatedEmailsFile()
8cdaf7f Introduce synchronized UserMap type
1b84eef Add UserMap methods, locking
af15dcf Reload authenticated-emails-file upon update
6d95548 Make UserMap operations lock-free
Per:
- http://stackoverflow.com/questions/21447463/is-assigning-a-pointer-atomic-in-golang
- https://groups.google.com/forum/#!msg/golang-nuts/ueSvaEKgyLY/ZW_74IC4PekJ
75755d5 Fix tests on Windows
d0eab2e Ignore email file watcher Chmod events
0b9798b Fix watcher on Ubuntu 12.04
3a8251a WaitForReplacement() to retry emails file watch
a57fd29 Add deferred File.Close() calls where needed
Because correctness: Don't leak file handles anywhere, and prepare for
future panics and early returns.
52ed3fd Log error and return if emails file doesn't parse
40100d4 Add gopkg.in/fsnotify.v1 dependency to Godeps file
17dfbbc Avoid a race when Remove is preceded by Chmod
2015-05-09 19:31:38 -04:00
type ValidatorTest struct {
2019-10-18 08:49:33 -07:00
authEmailFileName string
done chan bool
updateSeen bool
Reload authenticated-emails-file upon update
This change extracts the UserMap class from NewValidator() so that its
LoadAuthenticatedEmailsFile() method can be called concurrently. This method
is called by a goroutine containing a fsnotify.Watcher watching the
authenticated emails file.
Watching isn't forever aborted when the authenticated emails file disappears.
The goroutine will call os.Stat() up to twenty times a second if the file is
persistently missing, but that's the pathological case, not the common one.
The common case is that some editors (including Vim) will perform a
rename-and-replace when updating a file, triggering fsnotify.Rename events,
and the file will temporarily disappear. This watcher goroutine handles that
case.
Also, on some platforms (notably Arch Linux), a remove will be preceded by a
fsnotify.Chmod, causing a race between the upcoming fsnotify.Remove and the
call to UserMap.LoadAuthenticatedEmailsFile(). Hence, we treat fsnotify.Chmod
the same as fsnotify.Remove and fsnotify.Rename. There's no significant
penalty to re-adding a file to the watcher.
Also contains the following small changes from the summary of commits below:
- Minor optimization of email domain search
- Fixed api_test.go on Windows
- Add deferred File.Close() calls where needed
- Log error and return if emails file doesn't parse
These are the original commits from #89 squashed into this one:
0c6f2b6 Refactor validator_test to prepare for more tests
e0c792b Add more test cases to validator_test
a9a9d93 Minor optimization of email domain search
b763ea5 Extract LoadAuthenticatedEmailsFile()
8cdaf7f Introduce synchronized UserMap type
1b84eef Add UserMap methods, locking
af15dcf Reload authenticated-emails-file upon update
6d95548 Make UserMap operations lock-free
Per:
- http://stackoverflow.com/questions/21447463/is-assigning-a-pointer-atomic-in-golang
- https://groups.google.com/forum/#!msg/golang-nuts/ueSvaEKgyLY/ZW_74IC4PekJ
75755d5 Fix tests on Windows
d0eab2e Ignore email file watcher Chmod events
0b9798b Fix watcher on Ubuntu 12.04
3a8251a WaitForReplacement() to retry emails file watch
a57fd29 Add deferred File.Close() calls where needed
Because correctness: Don't leak file handles anywhere, and prepare for
future panics and early returns.
52ed3fd Log error and return if emails file doesn't parse
40100d4 Add gopkg.in/fsnotify.v1 dependency to Godeps file
17dfbbc Avoid a race when Remove is preceded by Chmod
2015-05-09 19:31:38 -04:00
}
func NewValidatorTest ( t * testing . T ) * ValidatorTest {
vt := & ValidatorTest { }
var err error
2022-10-21 12:57:51 +02:00
f , err := os . CreateTemp ( "" , "test_auth_emails_" )
2015-04-07 17:25:46 -04:00
if err != nil {
2019-10-18 08:49:33 -07:00
t . Fatalf ( "failed to create temp file: %v" , err )
2015-04-07 17:25:46 -04:00
}
2019-10-18 08:49:33 -07:00
if err := f . Close ( ) ; err != nil {
t . Fatalf ( "failed to close temp file: %v" , err )
}
vt . authEmailFileName = f . Name ( )
2017-10-31 09:12:15 -04:00
vt . done = make ( chan bool , 1 )
Reload authenticated-emails-file upon update
This change extracts the UserMap class from NewValidator() so that its
LoadAuthenticatedEmailsFile() method can be called concurrently. This method
is called by a goroutine containing a fsnotify.Watcher watching the
authenticated emails file.
Watching isn't forever aborted when the authenticated emails file disappears.
The goroutine will call os.Stat() up to twenty times a second if the file is
persistently missing, but that's the pathological case, not the common one.
The common case is that some editors (including Vim) will perform a
rename-and-replace when updating a file, triggering fsnotify.Rename events,
and the file will temporarily disappear. This watcher goroutine handles that
case.
Also, on some platforms (notably Arch Linux), a remove will be preceded by a
fsnotify.Chmod, causing a race between the upcoming fsnotify.Remove and the
call to UserMap.LoadAuthenticatedEmailsFile(). Hence, we treat fsnotify.Chmod
the same as fsnotify.Remove and fsnotify.Rename. There's no significant
penalty to re-adding a file to the watcher.
Also contains the following small changes from the summary of commits below:
- Minor optimization of email domain search
- Fixed api_test.go on Windows
- Add deferred File.Close() calls where needed
- Log error and return if emails file doesn't parse
These are the original commits from #89 squashed into this one:
0c6f2b6 Refactor validator_test to prepare for more tests
e0c792b Add more test cases to validator_test
a9a9d93 Minor optimization of email domain search
b763ea5 Extract LoadAuthenticatedEmailsFile()
8cdaf7f Introduce synchronized UserMap type
1b84eef Add UserMap methods, locking
af15dcf Reload authenticated-emails-file upon update
6d95548 Make UserMap operations lock-free
Per:
- http://stackoverflow.com/questions/21447463/is-assigning-a-pointer-atomic-in-golang
- https://groups.google.com/forum/#!msg/golang-nuts/ueSvaEKgyLY/ZW_74IC4PekJ
75755d5 Fix tests on Windows
d0eab2e Ignore email file watcher Chmod events
0b9798b Fix watcher on Ubuntu 12.04
3a8251a WaitForReplacement() to retry emails file watch
a57fd29 Add deferred File.Close() calls where needed
Because correctness: Don't leak file handles anywhere, and prepare for
future panics and early returns.
52ed3fd Log error and return if emails file doesn't parse
40100d4 Add gopkg.in/fsnotify.v1 dependency to Godeps file
17dfbbc Avoid a race when Remove is preceded by Chmod
2015-05-09 19:31:38 -04:00
return vt
}
2015-04-07 17:25:46 -04:00
Reload authenticated-emails-file upon update
This change extracts the UserMap class from NewValidator() so that its
LoadAuthenticatedEmailsFile() method can be called concurrently. This method
is called by a goroutine containing a fsnotify.Watcher watching the
authenticated emails file.
Watching isn't forever aborted when the authenticated emails file disappears.
The goroutine will call os.Stat() up to twenty times a second if the file is
persistently missing, but that's the pathological case, not the common one.
The common case is that some editors (including Vim) will perform a
rename-and-replace when updating a file, triggering fsnotify.Rename events,
and the file will temporarily disappear. This watcher goroutine handles that
case.
Also, on some platforms (notably Arch Linux), a remove will be preceded by a
fsnotify.Chmod, causing a race between the upcoming fsnotify.Remove and the
call to UserMap.LoadAuthenticatedEmailsFile(). Hence, we treat fsnotify.Chmod
the same as fsnotify.Remove and fsnotify.Rename. There's no significant
penalty to re-adding a file to the watcher.
Also contains the following small changes from the summary of commits below:
- Minor optimization of email domain search
- Fixed api_test.go on Windows
- Add deferred File.Close() calls where needed
- Log error and return if emails file doesn't parse
These are the original commits from #89 squashed into this one:
0c6f2b6 Refactor validator_test to prepare for more tests
e0c792b Add more test cases to validator_test
a9a9d93 Minor optimization of email domain search
b763ea5 Extract LoadAuthenticatedEmailsFile()
8cdaf7f Introduce synchronized UserMap type
1b84eef Add UserMap methods, locking
af15dcf Reload authenticated-emails-file upon update
6d95548 Make UserMap operations lock-free
Per:
- http://stackoverflow.com/questions/21447463/is-assigning-a-pointer-atomic-in-golang
- https://groups.google.com/forum/#!msg/golang-nuts/ueSvaEKgyLY/ZW_74IC4PekJ
75755d5 Fix tests on Windows
d0eab2e Ignore email file watcher Chmod events
0b9798b Fix watcher on Ubuntu 12.04
3a8251a WaitForReplacement() to retry emails file watch
a57fd29 Add deferred File.Close() calls where needed
Because correctness: Don't leak file handles anywhere, and prepare for
future panics and early returns.
52ed3fd Log error and return if emails file doesn't parse
40100d4 Add gopkg.in/fsnotify.v1 dependency to Godeps file
17dfbbc Avoid a race when Remove is preceded by Chmod
2015-05-09 19:31:38 -04:00
func ( vt * ValidatorTest ) TearDown ( ) {
2015-05-13 17:30:19 -04:00
vt . done <- true
2019-10-18 08:49:33 -07:00
os . Remove ( vt . authEmailFileName )
Reload authenticated-emails-file upon update
This change extracts the UserMap class from NewValidator() so that its
LoadAuthenticatedEmailsFile() method can be called concurrently. This method
is called by a goroutine containing a fsnotify.Watcher watching the
authenticated emails file.
Watching isn't forever aborted when the authenticated emails file disappears.
The goroutine will call os.Stat() up to twenty times a second if the file is
persistently missing, but that's the pathological case, not the common one.
The common case is that some editors (including Vim) will perform a
rename-and-replace when updating a file, triggering fsnotify.Rename events,
and the file will temporarily disappear. This watcher goroutine handles that
case.
Also, on some platforms (notably Arch Linux), a remove will be preceded by a
fsnotify.Chmod, causing a race between the upcoming fsnotify.Remove and the
call to UserMap.LoadAuthenticatedEmailsFile(). Hence, we treat fsnotify.Chmod
the same as fsnotify.Remove and fsnotify.Rename. There's no significant
penalty to re-adding a file to the watcher.
Also contains the following small changes from the summary of commits below:
- Minor optimization of email domain search
- Fixed api_test.go on Windows
- Add deferred File.Close() calls where needed
- Log error and return if emails file doesn't parse
These are the original commits from #89 squashed into this one:
0c6f2b6 Refactor validator_test to prepare for more tests
e0c792b Add more test cases to validator_test
a9a9d93 Minor optimization of email domain search
b763ea5 Extract LoadAuthenticatedEmailsFile()
8cdaf7f Introduce synchronized UserMap type
1b84eef Add UserMap methods, locking
af15dcf Reload authenticated-emails-file upon update
6d95548 Make UserMap operations lock-free
Per:
- http://stackoverflow.com/questions/21447463/is-assigning-a-pointer-atomic-in-golang
- https://groups.google.com/forum/#!msg/golang-nuts/ueSvaEKgyLY/ZW_74IC4PekJ
75755d5 Fix tests on Windows
d0eab2e Ignore email file watcher Chmod events
0b9798b Fix watcher on Ubuntu 12.04
3a8251a WaitForReplacement() to retry emails file watch
a57fd29 Add deferred File.Close() calls where needed
Because correctness: Don't leak file handles anywhere, and prepare for
future panics and early returns.
52ed3fd Log error and return if emails file doesn't parse
40100d4 Add gopkg.in/fsnotify.v1 dependency to Godeps file
17dfbbc Avoid a race when Remove is preceded by Chmod
2015-05-09 19:31:38 -04:00
}
2015-05-13 17:30:19 -04:00
func ( vt * ValidatorTest ) NewValidator ( domains [ ] string ,
updated chan <- bool ) func ( string ) bool {
2019-10-18 08:49:33 -07:00
return newValidatorImpl ( domains , vt . authEmailFileName ,
2015-05-13 18:30:22 -04:00
vt . done , func ( ) {
2018-11-29 14:26:41 +00:00
if vt . updateSeen == false {
2015-05-13 18:30:22 -04:00
updated <- true
2018-11-29 14:26:41 +00:00
vt . updateSeen = true
2015-05-13 18:30:22 -04:00
}
} )
2015-05-13 17:30:19 -04:00
}
Reload authenticated-emails-file upon update
This change extracts the UserMap class from NewValidator() so that its
LoadAuthenticatedEmailsFile() method can be called concurrently. This method
is called by a goroutine containing a fsnotify.Watcher watching the
authenticated emails file.
Watching isn't forever aborted when the authenticated emails file disappears.
The goroutine will call os.Stat() up to twenty times a second if the file is
persistently missing, but that's the pathological case, not the common one.
The common case is that some editors (including Vim) will perform a
rename-and-replace when updating a file, triggering fsnotify.Rename events,
and the file will temporarily disappear. This watcher goroutine handles that
case.
Also, on some platforms (notably Arch Linux), a remove will be preceded by a
fsnotify.Chmod, causing a race between the upcoming fsnotify.Remove and the
call to UserMap.LoadAuthenticatedEmailsFile(). Hence, we treat fsnotify.Chmod
the same as fsnotify.Remove and fsnotify.Rename. There's no significant
penalty to re-adding a file to the watcher.
Also contains the following small changes from the summary of commits below:
- Minor optimization of email domain search
- Fixed api_test.go on Windows
- Add deferred File.Close() calls where needed
- Log error and return if emails file doesn't parse
These are the original commits from #89 squashed into this one:
0c6f2b6 Refactor validator_test to prepare for more tests
e0c792b Add more test cases to validator_test
a9a9d93 Minor optimization of email domain search
b763ea5 Extract LoadAuthenticatedEmailsFile()
8cdaf7f Introduce synchronized UserMap type
1b84eef Add UserMap methods, locking
af15dcf Reload authenticated-emails-file upon update
6d95548 Make UserMap operations lock-free
Per:
- http://stackoverflow.com/questions/21447463/is-assigning-a-pointer-atomic-in-golang
- https://groups.google.com/forum/#!msg/golang-nuts/ueSvaEKgyLY/ZW_74IC4PekJ
75755d5 Fix tests on Windows
d0eab2e Ignore email file watcher Chmod events
0b9798b Fix watcher on Ubuntu 12.04
3a8251a WaitForReplacement() to retry emails file watch
a57fd29 Add deferred File.Close() calls where needed
Because correctness: Don't leak file handles anywhere, and prepare for
future panics and early returns.
52ed3fd Log error and return if emails file doesn't parse
40100d4 Add gopkg.in/fsnotify.v1 dependency to Godeps file
17dfbbc Avoid a race when Remove is preceded by Chmod
2015-05-09 19:31:38 -04:00
func ( vt * ValidatorTest ) WriteEmails ( t * testing . T , emails [ ] string ) {
2019-10-18 08:49:33 -07:00
f , err := os . OpenFile ( vt . authEmailFileName , os . O_WRONLY , 0600 )
if err != nil {
t . Fatalf ( "failed to open auth email file: %v" , err )
}
if _ , err := f . WriteString ( strings . Join ( emails , "\n" ) ) ; err != nil {
t . Fatalf ( "failed to write emails to auth email file: %v" , err )
}
if err := f . Close ( ) ; err != nil {
t . Fatalf ( "failed to close auth email file: %v" , err )
Reload authenticated-emails-file upon update
This change extracts the UserMap class from NewValidator() so that its
LoadAuthenticatedEmailsFile() method can be called concurrently. This method
is called by a goroutine containing a fsnotify.Watcher watching the
authenticated emails file.
Watching isn't forever aborted when the authenticated emails file disappears.
The goroutine will call os.Stat() up to twenty times a second if the file is
persistently missing, but that's the pathological case, not the common one.
The common case is that some editors (including Vim) will perform a
rename-and-replace when updating a file, triggering fsnotify.Rename events,
and the file will temporarily disappear. This watcher goroutine handles that
case.
Also, on some platforms (notably Arch Linux), a remove will be preceded by a
fsnotify.Chmod, causing a race between the upcoming fsnotify.Remove and the
call to UserMap.LoadAuthenticatedEmailsFile(). Hence, we treat fsnotify.Chmod
the same as fsnotify.Remove and fsnotify.Rename. There's no significant
penalty to re-adding a file to the watcher.
Also contains the following small changes from the summary of commits below:
- Minor optimization of email domain search
- Fixed api_test.go on Windows
- Add deferred File.Close() calls where needed
- Log error and return if emails file doesn't parse
These are the original commits from #89 squashed into this one:
0c6f2b6 Refactor validator_test to prepare for more tests
e0c792b Add more test cases to validator_test
a9a9d93 Minor optimization of email domain search
b763ea5 Extract LoadAuthenticatedEmailsFile()
8cdaf7f Introduce synchronized UserMap type
1b84eef Add UserMap methods, locking
af15dcf Reload authenticated-emails-file upon update
6d95548 Make UserMap operations lock-free
Per:
- http://stackoverflow.com/questions/21447463/is-assigning-a-pointer-atomic-in-golang
- https://groups.google.com/forum/#!msg/golang-nuts/ueSvaEKgyLY/ZW_74IC4PekJ
75755d5 Fix tests on Windows
d0eab2e Ignore email file watcher Chmod events
0b9798b Fix watcher on Ubuntu 12.04
3a8251a WaitForReplacement() to retry emails file watch
a57fd29 Add deferred File.Close() calls where needed
Because correctness: Don't leak file handles anywhere, and prepare for
future panics and early returns.
52ed3fd Log error and return if emails file doesn't parse
40100d4 Add gopkg.in/fsnotify.v1 dependency to Godeps file
17dfbbc Avoid a race when Remove is preceded by Chmod
2015-05-09 19:31:38 -04:00
}
}
2021-06-29 20:37:03 +02:00
func TestValidatorOverwriteEmailListDirectly ( t * testing . T ) {
testCasesPreUpdate := [ ] struct {
name string
email string
expectedAuthZ bool
} {
{
name : "FirstEmailInList" ,
email : "xyzzy@example.com" ,
expectedAuthZ : true ,
} ,
{
name : "SecondEmailInList" ,
email : "plugh@example.com" ,
expectedAuthZ : true ,
} ,
{
name : "EmailNotInListThatMatchesNoDomains" ,
email : "xyzzy.plugh@example.com" ,
expectedAuthZ : false ,
} ,
2015-04-07 17:25:46 -04:00
}
2021-06-29 20:37:03 +02:00
testCasesPostUpdate := [ ] struct {
name string
email string
expectedAuthZ bool
} {
{
name : "email removed from list" ,
email : "xyzzy@example.com" ,
expectedAuthZ : false ,
} ,
{
name : "email retained in list" ,
email : "plugh@example.com" ,
expectedAuthZ : true ,
} ,
{
name : "email added to list" ,
email : "xyzzy.plugh@example.com" ,
expectedAuthZ : true ,
} ,
2015-12-23 00:08:51 -08:00
}
2019-10-18 08:49:33 -07:00
vt := NewValidatorTest ( t )
defer vt . TearDown ( )
vt . WriteEmails ( t , [ ] string {
"xyzzy@example.com" ,
"plugh@example.com" ,
} )
updated := make ( chan bool )
2021-06-29 20:37:03 +02:00
validator := vt . NewValidator ( [ ] string ( nil ) , updated )
2019-10-18 08:49:33 -07:00
2021-06-29 20:37:03 +02:00
for _ , tc := range testCasesPreUpdate {
t . Run ( tc . name , func ( t * testing . T ) {
g := NewWithT ( t )
authorized := validator ( tc . email )
g . Expect ( authorized ) . To ( Equal ( tc . expectedAuthZ ) )
} )
2019-10-18 08:49:33 -07:00
}
vt . WriteEmails ( t , [ ] string {
"xyzzy.plugh@example.com" ,
"plugh@example.com" ,
} )
<- updated
2021-06-29 20:37:03 +02:00
for _ , tc := range testCasesPostUpdate {
t . Run ( tc . name , func ( t * testing . T ) {
g := NewWithT ( t )
authorized := validator ( tc . email )
g . Expect ( authorized ) . To ( Equal ( tc . expectedAuthZ ) )
} )
2019-10-18 08:49:33 -07:00
}
2021-06-29 20:37:03 +02:00
}
func TestValidatorCases ( t * testing . T ) {
testCases := [ ] struct {
name string
allowedEmails [ ] string
allowedDomains [ ] string
email string
expectedAuthZ bool
} {
{
name : "EmailNotInCorrect1stSubDomainsNotInEmails" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { ".example0.com" , ".example1.com" } ,
email : "foo.bar@example0.com" ,
expectedAuthZ : false ,
} ,
2021-07-28 10:12:00 +02:00
{
name : "EmailNotInCorrect1stSubDomainsNotInEmailsWildcard" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { "*.example0.com" , "*.example1.com" } ,
email : "foo.bar@example0.com" ,
expectedAuthZ : false ,
} ,
2021-06-29 20:37:03 +02:00
{
name : "EmailInFirstDomain" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { ".example0.com" , ".example1.com" } ,
email : "foo@bar.example0.com" ,
expectedAuthZ : true ,
} ,
2021-07-28 10:12:00 +02:00
{
name : "EmailInFirstDomainWildcard" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { "*.example0.com" , "*.example1.com" } ,
email : "foo@bar.example0.com" ,
expectedAuthZ : true ,
} ,
2021-06-29 20:37:03 +02:00
{
name : "EmailNotInCorrect2ndSubDomainsNotInEmails" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { ".example0.com" , ".example1.com" } ,
email : "baz.quux@example1.com" ,
expectedAuthZ : false ,
} ,
{
name : "EmailInSecondDomain" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { ".example0.com" , ".example1.com" } ,
email : "baz@quux.example1.com" ,
expectedAuthZ : true ,
} ,
2021-07-28 10:12:00 +02:00
{
name : "EmailInSecondDomainWildcard" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { "*.example0.com" , "*.example1.com" } ,
email : "baz@quux.example1.com" ,
expectedAuthZ : true ,
} ,
2021-06-29 20:37:03 +02:00
{
name : "EmailInFirstEmailList" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { ".example0.com" , ".example1.com" } ,
email : "xyzzy@example.com" ,
expectedAuthZ : true ,
} ,
2021-07-28 10:12:00 +02:00
{
name : "EmailInFirstEmailListWildcard" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { "*.example0.com" , "*.example1.com" } ,
email : "xyzzy@example.com" ,
expectedAuthZ : true ,
} ,
2021-06-29 20:37:03 +02:00
{
name : "EmailNotInDomainsNotInEmails" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { ".example0.com" , ".example1.com" } ,
email : "xyzzy.plugh@example.com" ,
expectedAuthZ : false ,
} ,
{
name : "EmailInLastEmailList" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { ".example0.com" , ".example1.com" } ,
email : "plugh@example.com" ,
expectedAuthZ : true ,
} ,
{
name : "EmailIn1stSubdomain" ,
allowedEmails : nil ,
allowedDomains : [ ] string { "us.example.com" , "de.example.com" , "example.com" } ,
email : "xyzzy@us.example.com" ,
expectedAuthZ : true ,
} ,
{
name : "EmailIn2ndSubdomain" ,
allowedEmails : nil ,
allowedDomains : [ ] string { "us.example.com" , "de.example.com" , "example.com" } ,
email : "xyzzy@de.example.com" ,
expectedAuthZ : true ,
} ,
{
name : "EmailNotInAnySubdomain" ,
allowedEmails : nil ,
allowedDomains : [ ] string { "us.example.com" , "de.example.com" , "example.com" } ,
email : "global@au.example.com" ,
expectedAuthZ : false ,
} ,
{
name : "EmailInLastSubdomain" ,
allowedEmails : nil ,
allowedDomains : [ ] string { "us.example.com" , "de.example.com" , "example.com" } ,
email : "xyzzy@example.com" ,
expectedAuthZ : true ,
} ,
{
name : "EmailDomainNotCompletelyMatch" ,
allowedEmails : nil ,
allowedDomains : [ ] string { ".example.com" , ".example1.com" } ,
email : "something@fooexample.com" ,
expectedAuthZ : false ,
} ,
{
name : "HackerExtraDomainPrefix1" ,
allowedEmails : nil ,
allowedDomains : [ ] string { ".mycompany.com" } ,
email : "something@evilhackmycompany.com" ,
expectedAuthZ : false ,
} ,
{
name : "HackerExtraDomainPrefix2" ,
allowedEmails : nil ,
allowedDomains : [ ] string { ".mycompany.com" } ,
email : "something@ext.evilhackmycompany.com" ,
expectedAuthZ : false ,
} ,
{
name : "EmptyDomainAndEmailList" ,
allowedEmails : [ ] string ( nil ) ,
allowedDomains : [ ] string ( nil ) ,
email : "foo.bar@example.com" ,
expectedAuthZ : false ,
} ,
{
name : "EmailMatchWithAllowedEmails" ,
email : "foo.bar@example.com" ,
allowedEmails : [ ] string { "foo.bar@example.com" } ,
allowedDomains : [ ] string { "example.com" } ,
expectedAuthZ : true ,
} ,
{
name : "EmailFromSameDomainButNotInList" ,
email : "baz.quux@example.com" ,
allowedEmails : [ ] string { "foo.bar@example.com" } ,
allowedDomains : [ ] string ( nil ) ,
expectedAuthZ : false ,
} ,
{
name : "EmailMatchOnDomain" ,
email : "foo.bar@example.com" ,
allowedEmails : [ ] string ( nil ) ,
allowedDomains : [ ] string { "example.com" } ,
expectedAuthZ : true ,
} ,
{
name : "EmailMatchOnDomain2" ,
email : "baz.quux@example.com" ,
allowedEmails : [ ] string ( nil ) ,
allowedDomains : [ ] string { "example.com" } ,
expectedAuthZ : true ,
} ,
{
name : "EmailFromFirstDomainShouldValidate" ,
email : "foo.bar@example0.com" ,
allowedEmails : [ ] string { "Foo.Bar@Example.Com" } ,
allowedDomains : [ ] string { "example0.com" , "example1.com" } ,
expectedAuthZ : true ,
} ,
{
name : "EmailFromSecondDomainShouldValidate" ,
email : "baz.quux@example1.com" ,
allowedEmails : [ ] string { "Foo.Bar@Example.Com" } ,
allowedDomains : [ ] string { "example0.com" , "example1.com" } ,
expectedAuthZ : true ,
} ,
{
name : "FirstEmailInListShouldValidate" ,
email : "xyzzy@example.com" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { "example0.com" , "example1.com" } ,
expectedAuthZ : true ,
} ,
{
name : "SecondEmailInListShouldValidate" ,
email : "plugh@example.com" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { "example0.com" , "example1.com" } ,
expectedAuthZ : true ,
} ,
{
name : "EmailNotInListThatMatchesNoDomains " ,
email : "xyzzy.plugh@example.com" ,
allowedEmails : [ ] string { "xyzzy@example.com" , "plugh@example.com" } ,
allowedDomains : [ ] string { "example0.com" , "example1.com" } ,
expectedAuthZ : false ,
} ,
{
name : "LoadedEmailAddressesAreNotLowerCased" ,
email : "foo.bar@example.com" ,
allowedEmails : [ ] string { "Foo.Bar@Example.Com" } ,
allowedDomains : [ ] string { "Frobozz.Com" } ,
expectedAuthZ : true ,
} ,
{
name : "ValidatedEmailAddressesAreNotLowerCased" ,
email : "Foo.Bar@Example.Com" ,
allowedEmails : [ ] string { "Foo.Bar@Example.Com" } ,
allowedDomains : [ ] string { "Frobozz.Com" } ,
expectedAuthZ : true ,
} ,
{
name : "LoadedDomainsAreNotLowerCased" ,
email : "foo.bar@frobozz.com" ,
allowedEmails : [ ] string { "Foo.Bar@Example.Com" } ,
allowedDomains : [ ] string { "Frobozz.Com" } ,
expectedAuthZ : true ,
} ,
{
name : "ValidatedDomainsAreNotLowerCased" ,
email : "foo.bar@Frobozz.Com" ,
allowedEmails : [ ] string { "Foo.Bar@Example.Com" } ,
allowedDomains : [ ] string { "Frobozz.Com" } ,
expectedAuthZ : true ,
} ,
{
name : "IgnoreSpacesInAuthEmails" ,
email : "foo.bar@example.com" ,
allowedEmails : [ ] string { " foo.bar@example.com " } ,
allowedDomains : [ ] string ( nil ) ,
expectedAuthZ : true ,
} ,
{
name : "IgnorePrefixSpacesInAuthEmails" ,
email : "foo.bar@example.com" ,
allowedEmails : [ ] string { " foo.bar@example.com" } ,
allowedDomains : [ ] string ( nil ) ,
expectedAuthZ : true ,
} ,
{
name : "CheckForEqualityNotSuffix" ,
email : "foo@evilcompany.com" ,
allowedEmails : [ ] string ( nil ) ,
allowedDomains : [ ] string { ".company.com" } ,
expectedAuthZ : false ,
} ,
{
name : "CheckForEqualityNotSuffix2" ,
email : "foo@evilcompany.com" ,
allowedEmails : [ ] string ( nil ) ,
allowedDomains : [ ] string { "company.com" } ,
expectedAuthZ : false ,
} ,
2021-07-28 10:12:00 +02:00
{
name : "CheckForEqualityNotSuffixWildcard" ,
email : "foo@evilcompany.com" ,
allowedEmails : [ ] string ( nil ) ,
allowedDomains : [ ] string { "*.company.com" } ,
expectedAuthZ : false ,
} ,
2019-10-18 08:49:33 -07:00
}
2021-06-29 20:37:03 +02:00
for _ , tc := range testCases {
t . Run ( tc . name , func ( t * testing . T ) {
vt := NewValidatorTest ( t )
defer vt . TearDown ( )
g := NewWithT ( t )
vt . WriteEmails ( t , tc . allowedEmails )
validator := vt . NewValidator ( tc . allowedDomains , nil )
authorized := validator ( tc . email )
g . Expect ( authorized ) . To ( Equal ( tc . expectedAuthZ ) )
} )
2019-10-18 08:49:33 -07:00
}
}