1 namespace System.ComponentModel.DataAnnotations {
3 using System.ComponentModel.DataAnnotations.Resources;
4 using System.Text.RegularExpressions;
6 [AttributeUsage(AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Parameter, AllowMultiple = false)]
7 public sealed class PhoneAttribute : DataTypeAttribute {
8 // see unit tests for examples
9 private static Regex _regex = CreateRegEx();
10 private const string _additionalPhoneNumberCharacters = "-.()";
12 public PhoneAttribute()
13 : base(DataType.PhoneNumber) {
15 // DevDiv 468241: set DefaultErrorMessage not ErrorMessage, allowing user to set
16 // ErrorMessageResourceType and ErrorMessageResourceName to use localized messages.
17 DefaultErrorMessage = DataAnnotationsResources.PhoneAttribute_Invalid;
20 public override bool IsValid(object value) {
25 string valueAsString = value as string;
27 // Use RegEx implementation if it has been created, otherwise use a non RegEx version.
29 return valueAsString != null && _regex.Match(valueAsString).Length > 0;
32 if (valueAsString == null) {
36 valueAsString = valueAsString.Replace("+", string.Empty).TrimEnd();
37 valueAsString = RemoveExtension(valueAsString);
39 bool digitFound = false;
40 foreach (char c in valueAsString) {
41 if (Char.IsDigit(c)) {
51 foreach (char c in valueAsString)
54 || Char.IsWhiteSpace(c)
55 || _additionalPhoneNumberCharacters.IndexOf(c) != -1)) {
63 private static Regex CreateRegEx() {
64 // We only need to create the RegEx if this switch is enabled.
65 if (AppSettings.DisableRegEx) {
69 const string pattern = @"^(\+\s?)?((?<!\+.*)\(\+?\d+([\s\-\.]?\d+)?\)|\d+)([\s\-\.]?(\(\d+([\s\-\.]?\d+)?\)|\d+))*(\s?(x|ext\.?)\s?\d+)?$";
70 const RegexOptions options = RegexOptions.Compiled | RegexOptions.IgnoreCase | RegexOptions.ExplicitCapture;
72 // Set explicit regex match timeout, sufficient enough for phone parsing
73 // Unless the global REGEX_DEFAULT_MATCH_TIMEOUT is already set
74 TimeSpan matchTimeout = TimeSpan.FromSeconds(2);
77 if (AppDomain.CurrentDomain.GetData("REGEX_DEFAULT_MATCH_TIMEOUT") == null) {
78 return new Regex(pattern, options, matchTimeout);
85 // Legacy fallback (without explicit match timeout)
86 return new Regex(pattern, options);
89 private static string RemoveExtension(string potentialPhoneNumber) {
90 int lastIndexOfExtension = potentialPhoneNumber
91 .LastIndexOf("ext.", StringComparison.InvariantCultureIgnoreCase);
92 if (lastIndexOfExtension >= 0) {
93 string extension = potentialPhoneNumber.Substring(lastIndexOfExtension + 4);
94 if (MatchesExtension(extension)) {
95 return potentialPhoneNumber.Substring(0, lastIndexOfExtension);
99 lastIndexOfExtension = potentialPhoneNumber
100 .LastIndexOf("ext", StringComparison.InvariantCultureIgnoreCase);
101 if (lastIndexOfExtension >= 0) {
102 string extension = potentialPhoneNumber.Substring(lastIndexOfExtension + 3);
103 if (MatchesExtension(extension)) {
104 return potentialPhoneNumber.Substring(0, lastIndexOfExtension);
109 lastIndexOfExtension = potentialPhoneNumber
110 .LastIndexOf("x", StringComparison.InvariantCultureIgnoreCase);
111 if (lastIndexOfExtension >= 0) {
112 string extension = potentialPhoneNumber.Substring(lastIndexOfExtension + 1);
113 if (MatchesExtension(extension)) {
114 return potentialPhoneNumber.Substring(0, lastIndexOfExtension);
118 return potentialPhoneNumber;
121 private static bool MatchesExtension(string potentialExtension) {
122 potentialExtension = potentialExtension.TrimStart();
123 if (potentialExtension.Length == 0) {
127 foreach (char c in potentialExtension) {
128 if (!Char.IsDigit(c)) {