// C o p y r i g h t © 2 0 2 3 R a n g e p r o o f P t y L t d . A l l r i g h t s r e s e r v e d .
import Foundation
import Quick
import Nimble
@ testable import SessionUtilitiesKit
class BencodeDecoderSpec : QuickSpec {
override class func spec ( ) {
// MARK: - B e n c o d e D e c o d e r
describe ( " BencodeDecoder " ) {
// MARK: - - - - s h o u l d d e c o d e a b a s i c s t r i n g
it ( " should decode a basic string " ) {
let basicStringData : Data = " 5:howdy " . data ( using : . utf8 ) !
let result = try ? BencodeDecoder ( ) . decode ( String . self , from : basicStringData )
expect ( result ) . to ( equal ( " howdy " ) )
}
// MARK: - - - - s h o u l d d e c o d e a b a s i c i n t e g e r
it ( " should decode a basic integer " ) {
let basicIntegerData : Data = " i3e " . data ( using : . utf8 ) !
let result = try ? BencodeDecoder ( ) . decode ( Int . self , from : basicIntegerData )
expect ( result ) . to ( equal ( 3 ) )
}
// MARK: - - - - s h o u l d d e c o d e a l i s t o f i n t e g e r s
it ( " should decode a list of integers " ) {
let basicIntListData : Data = " li1ei2ee " . data ( using : . utf8 ) !
let result = try ? BencodeDecoder ( ) . decode ( [ Int ] . self , from : basicIntListData )
expect ( result ) . to ( equal ( [ 1 , 2 ] ) )
}
// MARK: - - - - s h o u l d d e c o d e a b a s i c d i c t
it ( " should decode a basic dict " ) {
let basicDictData : Data = " d4:spaml1:a1:bee " . data ( using : . utf8 ) !
let result = try ? BencodeDecoder ( ) . decode ( [ String : [ String ] ] . self , from : basicDictData )
expect ( result ) . to ( equal ( [ " spam " : [ " a " , " b " ] ] ) )
}
// MARK: - - - - d e c o d e s a d e c o d a b l e t y p e
it ( " decodes a decodable type " ) {
let data : Data = " d8:intValuei100e11:stringValue4:Test " . data ( using : . utf8 ) !
let result : TestType ? = try ? BencodeDecoder ( ) . decode ( TestType . self , from : data )
expect ( result ) . to ( equal ( TestType ( intValue : 100 , stringValue : " Test " ) ) )
}
// MARK: - - - - - - t h r o w s a n e r r o r w h e n d e c o d i n g t h e w r o n g t y p e
it ( " throws an error when decoding the wrong type " ) {
let data : Data = " l4:Teste " . data ( using : . utf8 ) !
expect {
try BencodeDecoder ( ) . decode ( Int . self , from : data )
} . to ( throwError ( DecodingError . dataCorrupted ( DecodingError . Context ( codingPath : [ ] , debugDescription : " failed to decode String " ) ) ) )
}
// MARK: - - - - - - t h r o w s a n e r r o r w h e n g i v e n a n i n v a l i d l e n g t h
it ( " throws an error when given an invalid length " ) {
let data : Data = " d12:intValuei100e11:stringValue4:Teste "
. data ( using : . utf8 ) !
expect {
try BencodeDecoder ( ) . decode ( TestType . self , from : data )
} . to ( throwError ( DecodingError . keyNotFound ( TestType . CodingKeys . intValue , DecodingError . Context ( codingPath : [ ] , debugDescription : " key not found: intValue " ) ) ) )
}
// MARK: - - - - - - t h r o w s a n e r r o r w h e n g i v e n a n i n v a l i d k e y
it ( " throws an error when given an invalid key " ) {
let data : Data = " d7:INVALIDi100e11:stringValue4:Test "
. data ( using : . utf8 ) !
expect {
try BencodeDecoder ( ) . decode ( TestType . self , from : data )
} . to ( throwError ( DecodingError . keyNotFound ( TestType . CodingKeys . intValue , DecodingError . Context ( codingPath : [ ] , debugDescription : " key not found: intValue " ) ) ) )
}
// MARK: - - - - - - d e c o d e s c o r r e c t l y w h e n t r y i n g t o d e c o d e a n i n t t o a b o o l w i t h c u s t o m h a n d l i n g
it ( " decodes correctly when trying to decode an int to a bool with custom handling " ) {
let data : Data = " d9:boolValuei1e11:stringValue4:teste "
. data ( using : . utf8 ) !
expect {
try BencodeDecoder ( ) . decode ( TestType3 . self , from : data )
} . toNot ( throwError ( ) )
}
// MARK: - - - - - - t h r o w s a n e r r o r w h e n t r y i n g t o d e c o d e a n i n t t o a b o o l
it ( " throws an error when trying to decode an int to a bool " ) {
let data : Data = " d9:boolValuei1e11:stringValue4:teste "
. data ( using : . utf8 ) !
expect {
try BencodeDecoder ( ) . decode ( TestType2 . self , from : data )
} . to ( throwError ( DecodingError . typeMismatch ( Bool . self , DecodingError . Context ( codingPath : [ ] , debugDescription : " Bencode doesn't support Bool values, use an Int and custom Encode/Decode functions isntead " ) ) ) )
}
// MARK: - - - - d o e s n o t e n d u p i n a n i n f i n i t e l o o p w h e n d e c o d i n g I n t 6 4 t y p e s
it ( " does not end up in an infinite loop when decoding Int64 types " ) {
let basicIntListData : Data = " li1ei2ee " . data ( using : . utf8 ) !
let result = try ? BencodeDecoder ( ) . decode ( [ Int64 ] . self , from : basicIntListData )
expect ( result ) . to ( equal ( [ 1 , 2 ] ) )
}
// MARK: - - - - d o e s n o t e n d u p i n a n i n f i n i t e l o o p w h e n d e c o d i n g D o u b l e t y p e s
it ( " does not end up in an infinite loop when decoding Double types " ) {
let basicIntListData : Data = " li1ei2ee " . data ( using : . utf8 ) !
let result = try ? BencodeDecoder ( ) . decode ( [ Double ] . self , from : basicIntListData )
expect ( result ) . to ( equal ( [ 1 , 2 ] ) )
}
// MARK: - - - - d o e s n o t e n d u p i n a n i n f i n i t e l o o p w h e n d e c o d i n g F l o a t t y p e s
it ( " does not end up in an infinite loop when decoding Float types " ) {
let basicIntListData : Data = " li1ei2ee " . data ( using : . utf8 ) !
let result = try ? BencodeDecoder ( ) . decode ( [ Float ] . self , from : basicIntListData )
expect ( result ) . to ( equal ( [ 1 , 2 ] ) )
}
}
}
}
// MARK: - T e s t T y p e s
fileprivate struct TestType : Codable , Equatable {
public enum CodingKeys : String , CodingKey {
case intValue
case stringValue
}
let intValue : Int
let stringValue : String
}
fileprivate struct TestType2 : Codable , Equatable {
let stringValue : String
let boolValue : Bool
}
fileprivate struct TestType3 : Codable , Equatable {
let stringValue : String
let boolValue : Bool
init ( _ stringValue : String , _ boolValue : Bool ) {
self . stringValue = stringValue
self . boolValue = boolValue
}
init ( from decoder : Decoder ) throws {
let container : KeyedDecodingContainer < CodingKeys > = try decoder . container ( keyedBy : CodingKeys . self )
self = TestType3 (
try container . decode ( String . self , forKey : . stringValue ) ,
( ( try ? container . decode ( Bool . self , forKey : . boolValue ) ) ? ? false )
)
}
}