package matching import ( "strconv" "strings" "regexp" "github.com/nbutton23/zxcvbn-go/entropy" "github.com/nbutton23/zxcvbn-go/match" ) const ( dateSepMatcherName = "DATESEP" dateWithOutSepMatcherName = "DATEWITHOUT" ) var ( dateRxYearSuffix = regexp.MustCompile( `((\d{1,2})(\s|-|\/|\\|_|\.)(\d{1,2})(\s|-|\/|\\|_|\.)(19\d{2}|200\d|201\d|\d{2}))`) dateRxYearPrefix = regexp.MustCompile(`((19\d{2}|200\d|201\d|\d{2})(\s|-|/|\\|_|\.)(\d{1,2})(\s|-|/|\\|_|\.)(\d{1,2}))`) dateWithOutSepMatch = regexp.MustCompile( `\d{4,8}`) ) //FilterDateSepMatcher can be pass to zxcvbn-go.PasswordStrength to skip that matcher func FilterDateSepMatcher(m match.Matcher) bool { return m.ID == dateSepMatcherName } //FilterDateWithoutSepMatcher can be pass to zxcvbn-go.PasswordStrength to skip that matcher func FilterDateWithoutSepMatcher(m match.Matcher) bool { return m.ID == dateWithOutSepMatcherName } func checkDate(day, month, year int64) (bool, int64, int64, int64) { if (12 <= month && month <= 31) && day <= 12 { day, month = month, day } if day > 31 || month > 12 { return false, 0, 0, 0 } if !((1900 <= year && year <= 2019) || (0 <= year && year <= 99)) { return false, 0, 0, 0 } return true, day, month, year } func dateSepMatcher(password string) []match.Match { dateMatches := dateSepMatchHelper(password) var matches []match.Match for _, dateMatch := range dateMatches { match := match.Match{ I: dateMatch.I, J: dateMatch.J, Entropy: entropy.DateEntropy(dateMatch), DictionaryName: "date_match", Token: dateMatch.Token, } matches = append(matches, match) } return matches } func dateSepMatchHelper(password string) []match.DateMatch { var matches []match.DateMatch for _, v := range dateRxYearSuffix.FindAllString(password, len(password)) { splitV := dateRxYearSuffix.FindAllStringSubmatch(v, len(v)) i := strings.Index(password, v) j := i + len(v) day, _ := strconv.ParseInt(splitV[0][4], 10, 16) month, _ := strconv.ParseInt(splitV[0][2], 10, 16) year, _ := strconv.ParseInt(splitV[0][6], 10, 16) match := match.DateMatch{Day: day, Month: month, Year: year, Separator: splitV[0][5], I: i, J: j, Token: password[i:j]} matches = append(matches, match) } for _, v := range dateRxYearPrefix.FindAllString(password, len(password)) { splitV := dateRxYearPrefix.FindAllStringSubmatch(v, len(v)) i := strings.Index(password, v) j := i + len(v) day, _ := strconv.ParseInt(splitV[0][4], 10, 16) month, _ := strconv.ParseInt(splitV[0][6], 10, 16) year, _ := strconv.ParseInt(splitV[0][2], 10, 16) match := match.DateMatch{Day: day, Month: month, Year: year, Separator: splitV[0][5], I: i, J: j, Token: password[i:j]} matches = append(matches, match) } var out []match.DateMatch for _, match := range matches { if valid, day, month, year := checkDate(match.Day, match.Month, match.Year); valid { match.Pattern = "date" match.Day = day match.Month = month match.Year = year out = append(out, match) } } return out } type dateMatchCandidate struct { DayMonth string Year string I, J int } type dateMatchCandidateTwo struct { Day string Month string Year string I, J int } func dateWithoutSepMatch(password string) []match.Match { dateMatches := dateWithoutSepMatchHelper(password) var matches []match.Match for _, dateMatch := range dateMatches { match := match.Match{ I: dateMatch.I, J: dateMatch.J, Entropy: entropy.DateEntropy(dateMatch), DictionaryName: "date_match", Token: dateMatch.Token, } matches = append(matches, match) } return matches } //TODO Has issues with 6 digit dates func dateWithoutSepMatchHelper(password string) (matches []match.DateMatch) { for _, v := range dateWithOutSepMatch.FindAllString(password, len(password)) { i := strings.Index(password, v) j := i + len(v) length := len(v) lastIndex := length - 1 var candidatesRoundOne []dateMatchCandidate if length <= 6 { //2-digit year prefix candidatesRoundOne = append(candidatesRoundOne, buildDateMatchCandidate(v[2:], v[0:2], i, j)) //2-digityear suffix candidatesRoundOne = append(candidatesRoundOne, buildDateMatchCandidate(v[0:lastIndex-2], v[lastIndex-2:], i, j)) } if length >= 6 { //4-digit year prefix candidatesRoundOne = append(candidatesRoundOne, buildDateMatchCandidate(v[4:], v[0:4], i, j)) //4-digit year sufix candidatesRoundOne = append(candidatesRoundOne, buildDateMatchCandidate(v[0:lastIndex-3], v[lastIndex-3:], i, j)) } var candidatesRoundTwo []dateMatchCandidateTwo for _, c := range candidatesRoundOne { if len(c.DayMonth) == 2 { candidatesRoundTwo = append(candidatesRoundTwo, buildDateMatchCandidateTwo(c.DayMonth[0:0], c.DayMonth[1:1], c.Year, c.I, c.J)) } else if len(c.DayMonth) == 3 { candidatesRoundTwo = append(candidatesRoundTwo, buildDateMatchCandidateTwo(c.DayMonth[0:2], c.DayMonth[2:2], c.Year, c.I, c.J)) candidatesRoundTwo = append(candidatesRoundTwo, buildDateMatchCandidateTwo(c.DayMonth[0:0], c.DayMonth[1:3], c.Year, c.I, c.J)) } else if len(c.DayMonth) == 4 { candidatesRoundTwo = append(candidatesRoundTwo, buildDateMatchCandidateTwo(c.DayMonth[0:2], c.DayMonth[2:4], c.Year, c.I, c.J)) } } for _, candidate := range candidatesRoundTwo { intDay, err := strconv.ParseInt(candidate.Day, 10, 16) if err != nil { continue } intMonth, err := strconv.ParseInt(candidate.Month, 10, 16) if err != nil { continue } intYear, err := strconv.ParseInt(candidate.Year, 10, 16) if err != nil { continue } if ok, _, _, _ := checkDate(intDay, intMonth, intYear); ok { matches = append(matches, match.DateMatch{Token: password, Pattern: "date", Day: intDay, Month: intMonth, Year: intYear, I: i, J: j}) } } } return matches } func buildDateMatchCandidate(dayMonth, year string, i, j int) dateMatchCandidate { return dateMatchCandidate{DayMonth: dayMonth, Year: year, I: i, J: j} } func buildDateMatchCandidateTwo(day, month string, year string, i, j int) dateMatchCandidateTwo { return dateMatchCandidateTwo{Day: day, Month: month, Year: year, I: i, J: j} }