Skip to content

Commit 184553a

Browse files
committed
Add CreditCardValidationRule implementation
1 parent 4e03300 commit 184553a

1 file changed

Lines changed: 76 additions & 0 deletions

File tree

Lines changed: 76 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
//
2+
// Validator
3+
// Copyright © 2025 Space Code. All rights reserved.
4+
//
5+
6+
/// A credit card validation rule.
7+
public struct CreditCardValidationRule: IValidationRule {
8+
// MARK: Types
9+
10+
public enum CardType: String, Sendable, CaseIterable {
11+
case visa, masterCard, amex, jcb, unionPay
12+
}
13+
14+
public typealias Input = String
15+
16+
// MARK: Properties
17+
18+
public let types: [CardType]
19+
20+
/// The validation error.
21+
public let error: IValidationError
22+
23+
// MARK: Initialization
24+
25+
public init(types: [CardType] = CardType.allCases, error: IValidationError) {
26+
self.types = types
27+
self.error = error
28+
}
29+
30+
// MARK: IValidationRule
31+
32+
public func validate(input: String) -> Bool {
33+
let sanitized = input.replacingOccurrences(of: " ", with: "")
34+
35+
guard sanitized.allSatisfy(\.isNumber) else { return false }
36+
37+
guard types.contains(where: { matches(cardNumber: sanitized, type: $0) }) else { return false }
38+
39+
return isValidLuhn(sanitized)
40+
}
41+
42+
// MARK: Private
43+
44+
private func matches(cardNumber: String, type: CardType) -> Bool {
45+
switch type {
46+
case .visa:
47+
cardNumber.hasPrefix("4") && (cardNumber.count == 13 || cardNumber.count == 16 || cardNumber.count == 19)
48+
case .masterCard:
49+
(cardNumber.hasPrefix("51") || cardNumber.hasPrefix("52") ||
50+
cardNumber.hasPrefix("53") || cardNumber.hasPrefix("54") ||
51+
cardNumber.hasPrefix("55")) && cardNumber.count == 16
52+
case .amex:
53+
(cardNumber.hasPrefix("34") || cardNumber.hasPrefix("37")) && cardNumber.count == 15
54+
case .jcb:
55+
(cardNumber.hasPrefix("3528") || cardNumber.hasPrefix("3589")) && cardNumber.count == 16
56+
case .unionPay:
57+
cardNumber.hasPrefix("62") && (cardNumber.count >= 16 && cardNumber.count <= 19)
58+
}
59+
}
60+
61+
private func isValidLuhn(_ cardNumber: String) -> Bool {
62+
let reversedDigits = cardNumber.reversed().map { Int(String($0)) ?? 0 }
63+
var sum = 0
64+
65+
for (index, digit) in reversedDigits.enumerated() {
66+
if !index.isMultiple(of: 2) {
67+
let doubled = digit * 2
68+
sum += doubled > 9 ? doubled - 9 : doubled
69+
} else {
70+
sum += digit
71+
}
72+
}
73+
74+
return sum.isMultiple(of: 10)
75+
}
76+
}

0 commit comments

Comments
 (0)