-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathphone.go
170 lines (155 loc) · 5.55 KB
/
phone.go
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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
package is
import (
"fmt"
"regexp"
"strconv"
"github.com/goloop/g"
)
var (
// The e16Regex is regular expression for E.164.
e164Regex = regexp.MustCompile(`^\+\d{1,15}$`)
// The phoneRegex regular expression for phone number.
phoneRegex = regexp.MustCompile(`^\+[\d]+$`)
)
// IMSI checks if the given value is a valid International Mobile Subscriber
// Identity (IMSI). The IMSI is a unique identifier associated with a mobile
// network subscriber. It consists of three parts: MCC (Mobile Country Code),
// MNC (Mobile Network Code), and MSIN (Mobile Subscriber Identification
// Number).
//
// This function validates the IMSI by performing the following checks:
// 1. The length of the IMSI should be 15 digits.
// 2. The first three digits (MCC) should be a valid MCC.
// 3. The following two or three digits (MNC) should be a valid MNC.
// 4. The remaining digits (MSIN) should be numeric.
//
// Example usage:
//
// is.IMSI("310150123456789") // Returns: true
// is.IMSI("460001234567890") // Returns: true
// is.IMSI("1234567890123456") // Returns: false, length exceeds 15 digits
// is.IMSI("310150abc123456") // Returns: false, invalid characters in MSIN
func IMSI(imsi string) bool {
// Check the length of the IMSI.
if len(imsi) != 15 {
return false
}
// Check the first three digits (MCC).
mccStr := imsi[:3]
_, err := strconv.Atoi(mccStr)
if err != nil {
return false
}
// Check the next two or three digits (MNC).
mncStr := imsi[3:5]
_, err = strconv.Atoi(mncStr)
if err != nil {
return false
}
// Check the remaining digits (MSIN).
msinStr := imsi[5:]
if _, err := strconv.Atoi(msinStr); err != nil {
return false
}
return true
}
// IMEI validates whether a given string is a valid International Mobile
// Equipment Identity (IMEI). It uses the Luhn algorithm for verification.
//
// The value cannot contain characters other than numbers.
//
// The function takes as input a string 'v' representing a potential IMEI
// number, which could include spaces, dots, hyphens, or newline characters
// as separators. These are removed from the input string to obtain the raw
// number.
//
// The Luhn algorithm is then applied to this raw number to check its validity.
// This involves iterating over each digit in the number. If the digit is
// at an even-indexed position in the string (where the first position is
// considered index 1), its value is added directly to a running sum.
// If the digit is at an odd-indexed position, it is doubled.
// If the result of this doubling is greater than 9, 9 is subtracted from it.
// This resultant value is then added to the sum.
//
// After all digits have been processed, the function checks if the total
// sum is a multiple of 10. If it is, the function returns true, indicating
// that the input string is a valid IMEI.
// Otherwise, it returns false.
//
// Example usage:
//
// is.IMEI("522593572995861") // Returns: true
// is.IMEI("532593572995861") // Returns: false
//
// Note: An IMEI number is a unique identifier assigned to each mobile device.
// It is a 15-digit number used for tracking and identifying the device.
// The last digit is a check digit, computed according to the Luhn algorithm.
func IMEI[T string | int64](imei T) bool {
v := fmt.Sprint(imei)
if len(v) != 15 {
return false
}
sum := 0
for i := 0; i < len(v); i++ {
digit, _ := strconv.Atoi(string(v[i]))
// If the digit is in an even-indexed position (where the
// first position is 1), double its value.
if i%2 == 0 {
sum += digit
} else {
digit *= 2
if digit > 9 {
digit -= 9
}
sum += digit
}
}
// Check if the total sum is a multiple of 10.
// If it is, then the IMEI is valid.
return sum%10 == 0
}
// E164 checks if the string is a valid representation of a phone number
// according to the E.164 standard.
//
// The E.164 standard defines a numbering plan for the international
// public telecommunication numbering system. It specifies the format
// for international phone numbers and assigns a unique number to each
// country or region.
//
// This function uses a regular expression to match the input string against
// the E.164 phone number pattern. The pattern requires the string to start
// with a plus sign (+) followed by one or more digits. It allows a maximum
// length of 15 digits (excluding the plus sign).
//
// Example usage:
//
// is.E164("+123456789") // Returns: true
// is.E164("+0123456789") // Returns: true
// is.E164("+") // Returns: false, no digits after plus sign
// is.E164("+1234567890a") // Returns: false, non-digit character
// is.E164("1234567890") // Returns: false, no plus sign
// is.E164("") // Returns: false, empty string
//
// This function can be used to validate user input or data to ensure
// it follows the E.164 standard for phone numbers.
func E164(v string) bool {
return e164Regex.MatchString(v)
}
// Phone checks if the given string is a valid phone number.
// The phone number can have the following format:
// - It starts with a plus sign (+) followed by the country code.
// - The country code can be enclosed in parentheses.
// - The phone number can contain spaces between digits.
//
// Example usage:
//
// is.Phone("+380 (96) 00 00 000") // Returns: true
// is.Phone("+380961234567") // Returns: true
// is.Phone("123456789") // Returns: false, no plus sign
// is.Phone("") // Returns: false, empty string
//
// This function can be used to validate user input or data to ensure
// it follows the specified format for phone numbers.
func Phone(phone string) bool {
return phoneRegex.MatchString(g.Weed(phone, " ()"))
}