Skip to content

Commit 95025a4

Browse files
committed
Allow more than necessary padding characters
1 parent 1ea73d1 commit 95025a4

File tree

2 files changed

+49
-5
lines changed

2 files changed

+49
-5
lines changed

Sources/FoundationEssentials/Data/Data+Base64.swift

Lines changed: 21 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -523,11 +523,28 @@ extension Base64 {
523523
length: inout Int,
524524
options: Data.Base64DecodingOptions
525525
) throws(DecodingError) {
526-
let remaining = inBuffer.count % 4
527-
guard remaining == 0 else { throw DecodingError.invalidLength }
526+
guard let lastNonPaddedIndex = inBuffer.lastIndex(where: { $0 != UInt8(ascii: "=") }) else {
527+
if inBuffer.count >= 4 {
528+
outBuffer[0] = 0
529+
length = 1
530+
return
531+
} else {
532+
throw DecodingError.invalidLength
533+
}
534+
}
535+
let base64NonPaddedLength = lastNonPaddedIndex + 1
536+
let bytesToParseLength = if base64NonPaddedLength % 4 == 0 {
537+
(base64NonPaddedLength / 4) * 4
538+
} else {
539+
(base64NonPaddedLength / 4) * 4 + 4
540+
}
541+
if bytesToParseLength > inBuffer.count {
542+
throw DecodingError.invalidLength
543+
}
544+
545+
let outputLength = ((bytesToParseLength + 3) / 4) * 3
546+
let fullchunks = bytesToParseLength / 4 - 1
528547

529-
let outputLength = ((inBuffer.count + 3) / 4) * 3
530-
let fullchunks = remaining == 0 ? inBuffer.count / 4 - 1 : inBuffer.count / 4
531548
guard outBuffer.count >= outputLength else {
532549
preconditionFailure("Expected the out buffer to be at least as long as outputLength")
533550
}

Tests/FoundationEssentialsTests/DataTests.swift

Lines changed: 28 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2120,7 +2120,34 @@ extension DataTests {
21202120
func test_base64Data_bad() {
21212121
XCTAssertNil(Data(base64Encoded: "signature-not-base64-encoded"))
21222122
}
2123-
2123+
2124+
func test_base64Decode_MorePaddingThanNecessary() {
2125+
XCTAssertNil(Data(base64Encoded: "="))
2126+
XCTAssertNil(Data(base64Encoded: "=="))
2127+
XCTAssertNil(Data(base64Encoded: "==="))
2128+
for x in 4..<1000 {
2129+
XCTAssertEqual(Data(base64Encoded: String(repeating: "=", count: x)), Data([0]))
2130+
}
2131+
2132+
XCTAssertEqual(Data(base64Encoded: "AAAA"), Data([0, 0, 0]))
2133+
XCTAssertEqual(Data(base64Encoded: "AAAA="), Data([0, 0, 0]))
2134+
XCTAssertEqual(Data(base64Encoded: "AAAA=="), Data([0, 0, 0]))
2135+
XCTAssertEqual(Data(base64Encoded: "AAAA==="), Data([0, 0, 0]))
2136+
XCTAssertEqual(Data(base64Encoded: "AAAA===="), Data([0, 0, 0]))
2137+
XCTAssertEqual(Data(base64Encoded: "AAA="), Data([0, 0]))
2138+
XCTAssertEqual(Data(base64Encoded: "AAA=="), Data([0, 0]))
2139+
XCTAssertEqual(Data(base64Encoded: "AAA==="), Data([0, 0]))
2140+
XCTAssertEqual(Data(base64Encoded: "AAA===="), Data([0, 0]))
2141+
XCTAssertNil(Data(base64Encoded: "AA="))
2142+
XCTAssertEqual(Data(base64Encoded: "AA=="), Data([0]))
2143+
XCTAssertEqual(Data(base64Encoded: "AA==="), Data([0]))
2144+
XCTAssertEqual(Data(base64Encoded: "AA===="), Data([0]))
2145+
XCTAssertNil(Data(base64Encoded: "A="))
2146+
XCTAssertNil(Data(base64Encoded: "A=="))
2147+
XCTAssertNil(Data(base64Encoded: "A==="))
2148+
XCTAssertNil(Data(base64Encoded: "A===="))
2149+
}
2150+
21242151
func test_base64Data_medium() {
21252152
let data = Data("Lorem ipsum dolor sit amet, consectetur adipiscing elit. Ut at tincidunt arcu. Suspendisse nec sodales erat, sit amet imperdiet ipsum. Etiam sed ornare felis. Nunc mauris turpis, bibendum non lectus quis, malesuada placerat turpis. Nam adipiscing non massa et semper. Nulla convallis semper bibendum. Aliquam dictum nulla cursus mi ultricies, at tincidunt mi sagittis. Nulla faucibus at dui quis sodales. Morbi rutrum, dui id ultrices venenatis, arcu urna egestas felis, vel suscipit mauris arcu quis risus. Nunc venenatis ligula at orci tristique, et mattis purus pulvinar. Etiam ultricies est odio. Nunc eleifend malesuada justo, nec euismod sem ultrices quis. Etiam nec nibh sit amet lorem faucibus dapibus quis nec leo. Praesent sit amet mauris vel lacus hendrerit porta mollis consectetur mi. Donec eget tortor dui. Morbi imperdiet, arcu sit amet elementum interdum, quam nisl tempor quam, vitae feugiat augue purus sed lacus. In ac urna adipiscing purus venenatis volutpat vel et metus. Nullam nec auctor quam. Phasellus porttitor felis ac nibh gravida suscipit tempus at ante. Nunc pellentesque iaculis sapien a mattis. Aenean eleifend dolor non nunc laoreet, non dictum massa aliquam. Aenean quis turpis augue. Praesent augue lectus, mollis nec elementum eu, dignissim at velit. Ut congue neque id ullamcorper pellentesque. Maecenas euismod in elit eu vehicula. Nullam tristique dui nulla, nec convallis metus suscipit eget. Cras semper augue nec cursus blandit. Nulla rhoncus et odio quis blandit. Praesent lobortis dignissim velit ut pulvinar. Duis interdum quam adipiscing dolor semper semper. Nunc bibendum convallis dui, eget mollis magna hendrerit et. Morbi facilisis, augue eu fringilla convallis, mauris est cursus dolor, eu posuere odio nunc quis orci. Ut eu justo sem. Phasellus ut erat rhoncus, faucibus arcu vitae, vulputate erat. Aliquam nec magna viverra, interdum est vitae, rhoncus sapien. Duis tincidunt tempor ipsum ut dapibus. Nullam commodo varius metus, sed sollicitudin eros. Etiam nec odio et dui tempor blandit posuere.".utf8)
21262153
let base64 = data.base64EncodedString()

0 commit comments

Comments
 (0)