1 /** 2 * This module describes the digest APIs used in Phobos. All digests follow 3 * these APIs. Additionally, this module contains useful helper methods which 4 * can be used with every digest type. 5 * 6 $(SCRIPT inhibitQuickIndex = 1;) 7 8 $(DIVC quickindex, 9 $(BOOKTABLE , 10 $(TR $(TH Category) $(TH Functions) 11 ) 12 $(TR $(TDNW Template API) $(TD $(MYREF isDigest) $(MYREF DigestType) $(MYREF hasPeek) 13 $(MYREF hasBlockSize) 14 $(MYREF ExampleDigest) $(MYREF digest) $(MYREF hexDigest) $(MYREF makeDigest) 15 ) 16 ) 17 $(TR $(TDNW OOP API) $(TD $(MYREF Digest) 18 ) 19 ) 20 $(TR $(TDNW Helper functions) $(TD $(MYREF toHexString) $(MYREF secureEqual)) 21 ) 22 $(TR $(TDNW Implementation helpers) $(TD $(MYREF digestLength) $(MYREF WrapperDigest)) 23 ) 24 ) 25 ) 26 27 * APIs: 28 * There are two APIs for digests: The template API and the OOP API. The template API uses structs 29 * and template helpers like $(LREF isDigest). The OOP API implements digests as classes inheriting 30 * the $(LREF Digest) interface. All digests are named so that the template API struct is called "$(B x)" 31 * and the OOP API class is called "$(B x)Digest". For example we have `MD5` <--> `MD5Digest`, 32 * `CRC32` <--> `CRC32Digest`, etc. 33 * 34 * The template API is slightly more efficient. It does not have to allocate memory dynamically, 35 * all memory is allocated on the stack. The OOP API has to allocate in the finish method if no 36 * buffer was provided. If you provide a buffer to the OOP APIs finish function, it doesn't allocate, 37 * but the $(LREF Digest) classes still have to be created using `new` which allocates them using the GC. 38 * 39 * The OOP API is useful to change the digest function and/or digest backend at 'runtime'. The benefit here 40 * is that switching e.g. Phobos MD5Digest and an OpenSSLMD5Digest implementation is ABI compatible. 41 * 42 * If just one specific digest type and backend is needed, the template API is usually a good fit. 43 * In this simplest case, the template API can even be used without templates: Just use the "$(B x)" structs 44 * directly. 45 * 46 * License: $(HTTP www.boost.org/LICENSE_1_0.txt, Boost License 1.0). 47 * Authors: 48 * Johannes Pfau 49 * 50 * Source: $(PHOBOSSRC std/digest/package.d) 51 * 52 * CTFE: 53 * Digests do not work in CTFE 54 * 55 * TODO: 56 * Digesting single bits (as opposed to bytes) is not implemented. This will be done as another 57 * template constraint helper (hasBitDigesting!T) and an additional interface (BitDigest) 58 */ 59 /* Copyright Johannes Pfau 2012. 60 * Distributed under the Boost Software License, Version 1.0. 61 * (See accompanying file LICENSE_1_0.txt or copy at 62 * http://www.boost.org/LICENSE_1_0.txt) 63 */ 64 module std.digest; 65 66 public import std.ascii : LetterCase; 67 import std.meta : allSatisfy; 68 import std.range.primitives; 69 import std.traits; 70 71 72 /// 73 @system unittest 74 { 75 import std.digest.crc; 76 77 //Simple example 78 char[8] hexHash = hexDigest!CRC32("The quick brown fox jumps over the lazy dog"); 79 assert(hexHash == "39A34F41"); 80 81 //Simple example, using the API manually 82 CRC32 context = makeDigest!CRC32(); 83 context.put(cast(ubyte[])"The quick brown fox jumps over the lazy dog"); 84 ubyte[4] hash = context.finish(); 85 assert(toHexString(hash) == "39A34F41"); 86 } 87 88 /// 89 @system unittest 90 { 91 //Generating the hashes of a file, idiomatic D way 92 import std.digest.crc, std.digest.md, std.digest.sha; 93 import std.stdio; 94 95 // Digests a file and prints the result. 96 void digestFile(Hash)(string filename) 97 if (isDigest!Hash) 98 { 99 auto file = File(filename); 100 auto result = digest!Hash(file.byChunk(4096 * 1024)); 101 writefln("%s (%s) = %s", Hash.stringof, filename, toHexString(result)); 102 } 103 104 void main(string[] args) 105 { 106 foreach (name; args[1 .. $]) 107 { 108 digestFile!MD5(name); 109 digestFile!SHA1(name); 110 digestFile!CRC32(name); 111 } 112 } 113 } 114 /// 115 @system unittest 116 { 117 //Generating the hashes of a file using the template API 118 import std.digest.crc, std.digest.md, std.digest.sha; 119 import std.stdio; 120 // Digests a file and prints the result. 121 void digestFile(Hash)(ref Hash hash, string filename) 122 if (isDigest!Hash) 123 { 124 File file = File(filename); 125 126 //As digests imlement OutputRange, we could use std.algorithm.copy 127 //Let's do it manually for now 128 foreach (buffer; file.byChunk(4096 * 1024)) 129 hash.put(buffer); 130 131 auto result = hash.finish(); 132 writefln("%s (%s) = %s", Hash.stringof, filename, toHexString(result)); 133 } 134 135 void uMain(string[] args) 136 { 137 MD5 md5; 138 SHA1 sha1; 139 CRC32 crc32; 140 141 md5.start(); 142 sha1.start(); 143 crc32.start(); 144 145 foreach (arg; args[1 .. $]) 146 { 147 digestFile(md5, arg); 148 digestFile(sha1, arg); 149 digestFile(crc32, arg); 150 } 151 } 152 } 153 154 /// 155 @system unittest 156 { 157 import std.digest.crc, std.digest.md, std.digest.sha; 158 import std.stdio; 159 160 // Digests a file and prints the result. 161 void digestFile(Digest hash, string filename) 162 { 163 File file = File(filename); 164 165 //As digests implement OutputRange, we could use std.algorithm.copy 166 //Let's do it manually for now 167 foreach (buffer; file.byChunk(4096 * 1024)) 168 hash.put(buffer); 169 170 ubyte[] result = hash.finish(); 171 writefln("%s (%s) = %s", typeid(hash).toString(), filename, toHexString(result)); 172 } 173 174 void umain(string[] args) 175 { 176 auto md5 = new MD5Digest(); 177 auto sha1 = new SHA1Digest(); 178 auto crc32 = new CRC32Digest(); 179 180 foreach (arg; args[1 .. $]) 181 { 182 digestFile(md5, arg); 183 digestFile(sha1, arg); 184 digestFile(crc32, arg); 185 } 186 } 187 } 188 189 version (StdDdoc) 190 version = ExampleDigest; 191 192 version (ExampleDigest) 193 { 194 /** 195 * This documents the general structure of a Digest in the template API. 196 * All digest implementations should implement the following members and therefore pass 197 * the $(LREF isDigest) test. 198 * 199 * Note: 200 * $(UL 201 * $(LI A digest must be a struct (value type) to pass the $(LREF isDigest) test.) 202 * $(LI A digest passing the $(LREF isDigest) test is always an `OutputRange`) 203 * ) 204 */ 205 struct ExampleDigest 206 { 207 public: 208 /** 209 * Use this to feed the digest with data. 210 * Also implements the $(REF isOutputRange, std,range,primitives) 211 * interface for `ubyte` and `const(ubyte)[]`. 212 * The following usages of `put` must work for any type which 213 * passes $(LREF isDigest): 214 * Example: 215 * ---- 216 * ExampleDigest dig; 217 * dig.put(cast(ubyte) 0); //single ubyte 218 * dig.put(cast(ubyte) 0, cast(ubyte) 0); //variadic 219 * ubyte[10] buf; 220 * dig.put(buf); //buffer 221 * ---- 222 */ 223 @trusted void put(scope const(ubyte)[] data...) 224 { 225 226 } 227 228 /** 229 * This function is used to (re)initialize the digest. 230 * It must be called before using the digest and it also works as a 'reset' function 231 * if the digest has already processed data. 232 */ 233 @trusted void start() 234 { 235 236 } 237 238 /** 239 * The finish function returns the final hash sum and resets the Digest. 240 * 241 * Note: 242 * The actual type returned by finish depends on the digest implementation. 243 * `ubyte[16]` is just used as an example. It is guaranteed that the type is a 244 * static array of ubytes. 245 * 246 * $(UL 247 * $(LI Use $(LREF DigestType) to obtain the actual return type.) 248 * $(LI Use $(LREF digestLength) to obtain the length of the ubyte array.) 249 * ) 250 */ 251 @trusted ubyte[16] finish() 252 { 253 return (ubyte[16]).init; 254 } 255 } 256 } 257 258 /// 259 @system unittest 260 { 261 //Using the OutputRange feature 262 import std.algorithm.mutation : copy; 263 import std.digest.md; 264 import std.range : repeat; 265 266 auto oneMillionRange = repeat!ubyte(cast(ubyte)'a', 1000000); 267 auto ctx = makeDigest!MD5(); 268 copy(oneMillionRange, &ctx); //Note: You must pass a pointer to copy! 269 assert(ctx.finish().toHexString() == "7707D6AE4E027C70EEA2A935C2296F21"); 270 } 271 272 /** 273 * Use this to check if a type is a digest. See $(LREF ExampleDigest) to see what 274 * a type must provide to pass this check. 275 * 276 * Note: 277 * This is very useful as a template constraint (see examples) 278 * 279 * BUGS: 280 * $(UL 281 * $(LI Does not yet verify that put takes scope parameters.) 282 * $(LI Should check that finish() returns a ubyte[num] array) 283 * ) 284 */ 285 template isDigest(T) 286 { 287 import std.range : isOutputRange; 288 enum bool isDigest = isOutputRange!(T, const(ubyte)[]) && isOutputRange!(T, ubyte) && 289 is(T == struct) && 290 is(typeof( 291 { 292 T dig = void; //Can define 293 dig.put(cast(ubyte) 0, cast(ubyte) 0); //varags 294 dig.start(); //has start 295 auto value = dig.finish(); //has finish 296 })); 297 } 298 299 /// 300 @system unittest 301 { 302 import std.digest.crc; 303 static assert(isDigest!CRC32); 304 } 305 /// 306 @system unittest 307 { 308 import std.digest.crc; 309 void myFunction(T)() 310 if (isDigest!T) 311 { 312 T dig; 313 dig.start(); 314 auto result = dig.finish(); 315 } 316 myFunction!CRC32(); 317 } 318 319 /** 320 * Use this template to get the type which is returned by a digest's $(LREF finish) method. 321 */ 322 template DigestType(T) 323 { 324 static if (isDigest!T) 325 { 326 alias DigestType = 327 ReturnType!(typeof( 328 { 329 T dig = void; 330 return dig.finish(); 331 })); 332 } 333 else 334 static assert(false, T.stringof ~ " is not a digest! (fails isDigest!T)"); 335 } 336 337 /// 338 @system unittest 339 { 340 import std.digest.crc; 341 assert(is(DigestType!(CRC32) == ubyte[4])); 342 } 343 /// 344 @system unittest 345 { 346 import std.digest.crc; 347 CRC32 dig; 348 dig.start(); 349 DigestType!CRC32 result = dig.finish(); 350 } 351 352 /** 353 * Used to check if a digest supports the `peek` method. 354 * Peek has exactly the same function signatures as finish, but it doesn't reset 355 * the digest's internal state. 356 * 357 * Note: 358 * $(UL 359 * $(LI This is very useful as a template constraint (see examples)) 360 * $(LI This also checks if T passes $(LREF isDigest)) 361 * ) 362 */ 363 template hasPeek(T) 364 { 365 enum bool hasPeek = isDigest!T && 366 is(typeof( 367 { 368 T dig = void; //Can define 369 DigestType!T val = dig.peek(); 370 })); 371 } 372 373 /// 374 @system unittest 375 { 376 import std.digest.crc, std.digest.md; 377 assert(!hasPeek!(MD5)); 378 assert(hasPeek!CRC32); 379 } 380 /// 381 @system unittest 382 { 383 import std.digest.crc; 384 void myFunction(T)() 385 if (hasPeek!T) 386 { 387 T dig; 388 dig.start(); 389 auto result = dig.peek(); 390 } 391 myFunction!CRC32(); 392 } 393 394 /** 395 * Checks whether the digest has a `blockSize` member, which contains the 396 * digest's internal block size in bits. It is primarily used by $(REF HMAC, std,digest,hmac). 397 */ 398 399 template hasBlockSize(T) 400 if (isDigest!T) 401 { 402 enum bool hasBlockSize = __traits(compiles, { size_t blockSize = T.blockSize; }); 403 } 404 405 /// 406 @system unittest 407 { 408 import std.digest.hmac, std.digest.md; 409 static assert(hasBlockSize!MD5 && MD5.blockSize == 512); 410 static assert(hasBlockSize!(HMAC!MD5) && HMAC!MD5.blockSize == 512); 411 } 412 413 package template isDigestibleRange(Range) 414 { 415 import std.digest.md; 416 import std.range : isInputRange, ElementType; 417 enum bool isDigestibleRange = isInputRange!Range && is(typeof( 418 { 419 MD5 ha; //Could use any conformant hash 420 ElementType!Range val; 421 ha.put(val); 422 })); 423 } 424 425 /** 426 * This is a convenience function to calculate a hash using the template API. 427 * Every digest passing the $(LREF isDigest) test can be used with this function. 428 * 429 * Params: 430 * range= an `InputRange` with `ElementType` `ubyte`, `ubyte[]` or `ubyte[num]` 431 */ 432 DigestType!Hash digest(Hash, Range)(auto ref Range range) 433 if (!isArray!Range 434 && isDigestibleRange!Range) 435 { 436 Hash hash; 437 hash.start(); 438 alias E = ElementType!Range; // Not necessarily ubyte. Could be ubyte[N] or ubyte[] or something w/alias this. 439 static if (!(__traits(isScalar, E) && E.sizeof == 1)) 440 { 441 foreach (e; range) 442 hash.put(e); 443 return hash.finish(); 444 } 445 else 446 { 447 static if (hasBlockSize!Hash) 448 enum bufferBytes = Hash.blockSize >= (8192 * 8) ? 8192 : Hash.blockSize <= 64 ? 8 : (Hash.blockSize / 8); 449 else 450 enum bufferBytes = 8; 451 ubyte[bufferBytes] buffer = void; 452 static if (isRandomAccessRange!Range && hasLength!Range) 453 { 454 const end = range.length; 455 size_t i = 0; 456 while (end - i >= buffer.length) 457 { 458 foreach (ref e; buffer) 459 e = range[i++]; 460 hash.put(buffer); 461 } 462 if (const remaining = end - i) 463 { 464 foreach (ref e; buffer[0 .. remaining]) 465 e = range[i++]; 466 hash.put(buffer[0 .. remaining]); 467 } 468 return hash.finish(); 469 } 470 else 471 { 472 for (;;) 473 { 474 size_t n = buffer.length; 475 foreach (i, ref e; buffer) 476 { 477 if (range.empty) 478 { 479 n = i; 480 break; 481 } 482 e = range.front; 483 range.popFront(); 484 } 485 if (n) 486 hash.put(buffer[0 .. n]); 487 if (n != buffer.length) 488 return hash.finish(); 489 } 490 } 491 } 492 } 493 494 /// 495 @system unittest 496 { 497 import std.digest.md; 498 import std.range : repeat; 499 auto testRange = repeat!ubyte(cast(ubyte)'a', 100); 500 auto md5 = digest!MD5(testRange); 501 } 502 503 /** 504 * This overload of the digest function handles arrays. 505 * 506 * Params: 507 * data= one or more arrays of any type 508 */ 509 DigestType!Hash digest(Hash, T...)(scope const T data) 510 if (allSatisfy!(isArray, typeof(data))) 511 { 512 Hash hash; 513 hash.start(); 514 foreach (datum; data) 515 hash.put(cast(const(ubyte[]))datum); 516 return hash.finish(); 517 } 518 519 /// 520 @system unittest 521 { 522 import std.digest.crc, std.digest.md, std.digest.sha; 523 auto md5 = digest!MD5( "The quick brown fox jumps over the lazy dog"); 524 auto sha1 = digest!SHA1( "The quick brown fox jumps over the lazy dog"); 525 auto crc32 = digest!CRC32("The quick brown fox jumps over the lazy dog"); 526 assert(toHexString(crc32) == "39A34F41"); 527 } 528 529 /// 530 @system unittest 531 { 532 import std.digest.crc; 533 auto crc32 = digest!CRC32("The quick ", "brown ", "fox jumps over the lazy dog"); 534 assert(toHexString(crc32) == "39A34F41"); 535 } 536 537 /** 538 * This is a convenience function similar to $(LREF digest), but it returns the string 539 * representation of the hash. Every digest passing the $(LREF isDigest) test can be used with this 540 * function. 541 * 542 * Params: 543 * order= the order in which the bytes are processed (see $(LREF toHexString)) 544 * range= an `InputRange` with `ElementType` `ubyte`, `ubyte[]` or `ubyte[num]` 545 */ 546 char[digestLength!(Hash)*2] hexDigest(Hash, Order order = Order.increasing, Range)(ref Range range) 547 if (!isArray!Range && isDigestibleRange!Range) 548 { 549 return toHexString!order(digest!Hash(range)); 550 } 551 552 /// 553 @system unittest 554 { 555 import std.digest.md; 556 import std.range : repeat; 557 auto testRange = repeat!ubyte(cast(ubyte)'a', 100); 558 assert(hexDigest!MD5(testRange) == "36A92CC94A9E0FA21F625F8BFB007ADF"); 559 } 560 561 /** 562 * This overload of the hexDigest function handles arrays. 563 * 564 * Params: 565 * order= the order in which the bytes are processed (see $(LREF toHexString)) 566 * data= one or more arrays of any type 567 */ 568 char[digestLength!(Hash)*2] hexDigest(Hash, Order order = Order.increasing, T...)(scope const T data) 569 if (allSatisfy!(isArray, typeof(data))) 570 { 571 return toHexString!order(digest!Hash(data)); 572 } 573 574 /// 575 @system unittest 576 { 577 import std.digest.crc; 578 assert(hexDigest!(CRC32, Order.decreasing)("The quick brown fox jumps over the lazy dog") == "414FA339"); 579 } 580 /// 581 @system unittest 582 { 583 import std.digest.crc; 584 assert(hexDigest!(CRC32, Order.decreasing)("The quick ", "brown ", "fox jumps over the lazy dog") == "414FA339"); 585 } 586 587 /** 588 * This is a convenience function which returns an initialized digest, so it's not necessary to call 589 * start manually. 590 */ 591 Hash makeDigest(Hash)() 592 { 593 Hash hash; 594 hash.start(); 595 return hash; 596 } 597 598 /// 599 @system unittest 600 { 601 import std.digest.md; 602 auto md5 = makeDigest!MD5(); 603 md5.put(0); 604 assert(toHexString(md5.finish()) == "93B885ADFE0DA089CDF634904FD59F71"); 605 } 606 607 /*+*************************** End of template part, welcome to OOP land **************************/ 608 609 /** 610 * This describes the OOP API. To understand when to use the template API and when to use the OOP API, 611 * see the module documentation at the top of this page. 612 * 613 * The Digest interface is the base interface which is implemented by all digests. 614 * 615 * Note: 616 * A Digest implementation is always an `OutputRange` 617 */ 618 interface Digest 619 { 620 public: 621 /** 622 * Use this to feed the digest with data. 623 * Also implements the $(REF isOutputRange, std,range,primitives) 624 * interface for `ubyte` and `const(ubyte)[]`. 625 * 626 * Example: 627 * ---- 628 * void test(Digest dig) 629 * { 630 * dig.put(cast(ubyte) 0); //single ubyte 631 * dig.put(cast(ubyte) 0, cast(ubyte) 0); //variadic 632 * ubyte[10] buf; 633 * dig.put(buf); //buffer 634 * } 635 * ---- 636 */ 637 @trusted nothrow void put(scope const(ubyte)[] data...); 638 639 /** 640 * Resets the internal state of the digest. 641 * Note: 642 * $(LREF finish) calls this internally, so it's not necessary to call 643 * `reset` manually after a call to $(LREF finish). 644 */ 645 @trusted nothrow void reset(); 646 647 /** 648 * This is the length in bytes of the hash value which is returned by $(LREF finish). 649 * It's also the required size of a buffer passed to $(LREF finish). 650 */ 651 @trusted nothrow @property size_t length() const; 652 653 /** 654 * The finish function returns the hash value. It takes an optional buffer to copy the data 655 * into. If a buffer is passed, it must be at least $(LREF length) bytes big. 656 */ 657 @trusted nothrow ubyte[] finish(); 658 ///ditto 659 nothrow ubyte[] finish(ubyte[] buf); 660 // https://issues.dlang.org/show_bug.cgi?id=6549 661 /*in 662 { 663 assert(buf.length >= this.length); 664 }*/ 665 666 /** 667 * This is a convenience function to calculate the hash of a value using the OOP API. 668 */ 669 final @trusted nothrow ubyte[] digest(scope const(void[])[] data...) 670 { 671 this.reset(); 672 foreach (datum; data) 673 this.put(cast(ubyte[]) datum); 674 return this.finish(); 675 } 676 } 677 678 /// 679 @system unittest 680 { 681 //Using the OutputRange feature 682 import std.algorithm.mutation : copy; 683 import std.digest.md; 684 import std.range : repeat; 685 686 auto oneMillionRange = repeat!ubyte(cast(ubyte)'a', 1000000); 687 auto ctx = new MD5Digest(); 688 copy(oneMillionRange, ctx); 689 assert(ctx.finish().toHexString() == "7707D6AE4E027C70EEA2A935C2296F21"); 690 } 691 692 /// 693 @system unittest 694 { 695 import std.digest.crc, std.digest.md, std.digest.sha; 696 ubyte[] md5 = (new MD5Digest()).digest("The quick brown fox jumps over the lazy dog"); 697 ubyte[] sha1 = (new SHA1Digest()).digest("The quick brown fox jumps over the lazy dog"); 698 ubyte[] crc32 = (new CRC32Digest()).digest("The quick brown fox jumps over the lazy dog"); 699 assert(crcHexString(crc32) == "414FA339"); 700 } 701 702 /// 703 @system unittest 704 { 705 import std.digest.crc; 706 ubyte[] crc32 = (new CRC32Digest()).digest("The quick ", "brown ", "fox jumps over the lazy dog"); 707 assert(crcHexString(crc32) == "414FA339"); 708 } 709 710 @system unittest 711 { 712 import std.range : isOutputRange; 713 assert(!isDigest!(Digest)); 714 assert(isOutputRange!(Digest, ubyte)); 715 } 716 717 /// 718 @system unittest 719 { 720 void test(Digest dig) 721 { 722 dig.put(cast(ubyte) 0); //single ubyte 723 dig.put(cast(ubyte) 0, cast(ubyte) 0); //variadic 724 ubyte[10] buf; 725 dig.put(buf); //buffer 726 } 727 } 728 729 /*+*************************** End of OOP part, helper functions follow ***************************/ 730 731 /** 732 * See $(LREF toHexString) 733 */ 734 enum Order : bool 735 { 736 increasing, /// 737 decreasing /// 738 } 739 740 /// 741 @safe unittest 742 { 743 import std.digest.crc : CRC32; 744 745 auto crc32 = digest!CRC32("The quick ", "brown ", "fox jumps over the lazy dog"); 746 assert(crc32.toHexString!(Order.decreasing) == "414FA339"); 747 assert(crc32.toHexString!(LetterCase.lower, Order.decreasing) == "414fa339"); 748 } 749 750 751 /** 752 * Used to convert a hash value (a static or dynamic array of ubytes) to a string. 753 * Can be used with the OOP and with the template API. 754 * 755 * The additional order parameter can be used to specify the order of the input data. 756 * By default the data is processed in increasing order, starting at index 0. To process it in the 757 * opposite order, pass Order.decreasing as a parameter. 758 * 759 * The additional letterCase parameter can be used to specify the case of the output data. 760 * By default the output is in upper case. To change it to the lower case 761 * pass LetterCase.lower as a parameter. 762 * 763 * Note: 764 * The function overloads returning a string allocate their return values 765 * using the GC. The versions returning static arrays use pass-by-value for 766 * the return value, effectively avoiding dynamic allocation. 767 */ 768 char[num*2] toHexString(Order order = Order.increasing, size_t num, LetterCase letterCase = LetterCase.upper) 769 (const ubyte[num] digest) 770 { 771 772 char[num*2] result; 773 size_t i; 774 toHexStringImpl!(order, letterCase)(digest, result); 775 return result; 776 } 777 778 ///ditto 779 char[num*2] toHexString(LetterCase letterCase, Order order = Order.increasing, size_t num)(in ubyte[num] digest) 780 { 781 return toHexString!(order, num, letterCase)(digest); 782 } 783 784 ///ditto 785 string toHexString(Order order = Order.increasing, LetterCase letterCase = LetterCase.upper) 786 (in ubyte[] digest) 787 { 788 auto result = new char[digest.length*2]; 789 toHexStringImpl!(order, letterCase)(digest, result); 790 import std.exception : assumeUnique; 791 // memory was just created, so casting to immutable is safe 792 return () @trusted { return assumeUnique(result); }(); 793 } 794 795 ///ditto 796 string toHexString(LetterCase letterCase, Order order = Order.increasing)(in ubyte[] digest) 797 { 798 return toHexString!(order, letterCase)(digest); 799 } 800 801 //For more example unittests, see Digest.digest, digest 802 803 /// 804 @safe unittest 805 { 806 import std.digest.crc; 807 //Test with template API: 808 auto crc32 = digest!CRC32("The quick ", "brown ", "fox jumps over the lazy dog"); 809 //Lower case variant: 810 assert(toHexString!(LetterCase.lower)(crc32) == "39a34f41"); 811 //Usually CRCs are printed in this order, though: 812 assert(toHexString!(Order.decreasing)(crc32) == "414FA339"); 813 assert(toHexString!(LetterCase.lower, Order.decreasing)(crc32) == "414fa339"); 814 } 815 816 /// 817 @safe unittest 818 { 819 import std.digest.crc; 820 // With OOP API 821 auto crc32 = (new CRC32Digest()).digest("The quick ", "brown ", "fox jumps over the lazy dog"); 822 //Usually CRCs are printed in this order, though: 823 assert(toHexString!(Order.decreasing)(crc32) == "414FA339"); 824 } 825 826 @safe unittest 827 { 828 ubyte[16] data; 829 assert(toHexString(data) == "00000000000000000000000000000000"); 830 831 assert(toHexString(cast(ubyte[4])[42, 43, 44, 45]) == "2A2B2C2D"); 832 assert(toHexString(cast(ubyte[])[42, 43, 44, 45]) == "2A2B2C2D"); 833 assert(toHexString!(Order.decreasing)(cast(ubyte[4])[42, 43, 44, 45]) == "2D2C2B2A"); 834 assert(toHexString!(Order.decreasing, LetterCase.lower)(cast(ubyte[4])[42, 43, 44, 45]) == "2d2c2b2a"); 835 assert(toHexString!(Order.decreasing)(cast(ubyte[])[42, 43, 44, 45]) == "2D2C2B2A"); 836 } 837 838 /*+*********************** End of public helper part, private helpers follow ***********************/ 839 840 /* 841 * Used to convert from a ubyte[] slice to a ref ubyte[N]. 842 * This helper is used internally in the WrapperDigest template to wrap the template API's 843 * finish function. 844 */ 845 ref T[N] asArray(size_t N, T)(ref T[] source, string errorMsg = "") 846 { 847 assert(source.length >= N, errorMsg); 848 return *cast(T[N]*) source.ptr; 849 } 850 851 /* 852 * Fill in a preallocated buffer with the ASCII hex representation from a byte buffer 853 */ 854 private void toHexStringImpl(Order order, LetterCase letterCase, BB, HB) 855 (scope const ref BB byteBuffer, ref HB hexBuffer){ 856 static if (letterCase == LetterCase.upper) 857 { 858 import std.ascii : hexDigits = hexDigits; 859 } 860 else 861 { 862 import std.ascii : hexDigits = lowerHexDigits; 863 } 864 865 size_t i; 866 static if (order == Order.increasing) 867 { 868 foreach (u; byteBuffer) 869 { 870 hexBuffer[i++] = hexDigits[u >> 4]; 871 hexBuffer[i++] = hexDigits[u & 15]; 872 } 873 } 874 else 875 { 876 size_t j = byteBuffer.length -1; 877 while (i < byteBuffer.length*2) 878 { 879 hexBuffer[i++] = hexDigits[byteBuffer[j] >> 4]; 880 hexBuffer[i++] = hexDigits[byteBuffer[j] & 15]; 881 j--; 882 } 883 } 884 } 885 886 887 /* 888 * Returns the length (in bytes) of the hash value produced by T. 889 */ 890 template digestLength(T) 891 if (isDigest!T) 892 { 893 enum size_t digestLength = (ReturnType!(T.finish)).length; 894 } 895 896 @safe pure nothrow @nogc 897 unittest 898 { 899 import std.digest.md : MD5; 900 import std.digest.sha : SHA1, SHA256, SHA512; 901 assert(digestLength!MD5 == 16); 902 assert(digestLength!SHA1 == 20); 903 assert(digestLength!SHA256 == 32); 904 assert(digestLength!SHA512 == 64); 905 } 906 907 /** 908 * Wraps a template API hash struct into a Digest interface. 909 * Modules providing digest implementations will usually provide 910 * an alias for this template (e.g. MD5Digest, SHA1Digest, ...). 911 */ 912 class WrapperDigest(T) 913 if (isDigest!T) : Digest 914 { 915 protected: 916 T _digest; 917 918 public final: 919 /** 920 * Initializes the digest. 921 */ 922 this() 923 { 924 _digest.start(); 925 } 926 927 /** 928 * Use this to feed the digest with data. 929 * Also implements the $(REF isOutputRange, std,range,primitives) 930 * interface for `ubyte` and `const(ubyte)[]`. 931 */ 932 @trusted nothrow void put(scope const(ubyte)[] data...) 933 { 934 _digest.put(data); 935 } 936 937 /** 938 * Resets the internal state of the digest. 939 * Note: 940 * $(LREF finish) calls this internally, so it's not necessary to call 941 * `reset` manually after a call to $(LREF finish). 942 */ 943 @trusted nothrow void reset() 944 { 945 _digest.start(); 946 } 947 948 /** 949 * This is the length in bytes of the hash value which is returned by $(LREF finish). 950 * It's also the required size of a buffer passed to $(LREF finish). 951 */ 952 @trusted nothrow @property size_t length() const pure 953 { 954 return digestLength!T; 955 } 956 957 /** 958 * The finish function returns the hash value. It takes an optional buffer to copy the data 959 * into. If a buffer is passed, it must have a length at least $(LREF length) bytes. 960 * 961 * Example: 962 * -------- 963 * 964 * import std.digest.md; 965 * ubyte[16] buf; 966 * auto hash = new WrapperDigest!MD5(); 967 * hash.put(cast(ubyte) 0); 968 * auto result = hash.finish(buf[]); 969 * //The result is now in result (and in buf). If you pass a buffer which is bigger than 970 * //necessary, result will have the correct length, but buf will still have it's original 971 * //length 972 * -------- 973 */ 974 nothrow ubyte[] finish(ubyte[] buf) 975 in 976 { 977 assert(buf.length >= this.length, "Given buffer is smaller than the local buffer."); 978 } 979 do 980 { 981 enum string msg = "Buffer needs to be at least " ~ digestLength!(T).stringof ~ " bytes " ~ 982 "big, check " ~ typeof(this).stringof ~ ".length!"; 983 asArray!(digestLength!T)(buf, msg) = _digest.finish(); 984 return buf[0 .. digestLength!T]; 985 } 986 987 ///ditto 988 @trusted nothrow ubyte[] finish() 989 { 990 enum len = digestLength!T; 991 auto buf = new ubyte[len]; 992 asArray!(digestLength!T)(buf) = _digest.finish(); 993 return buf; 994 } 995 996 version (StdDdoc) 997 { 998 /** 999 * Works like `finish` but does not reset the internal state, so it's possible 1000 * to continue putting data into this WrapperDigest after a call to peek. 1001 * 1002 * These functions are only available if `hasPeek!T` is true. 1003 */ 1004 @trusted ubyte[] peek(ubyte[] buf) const; 1005 ///ditto 1006 @trusted ubyte[] peek() const; 1007 } 1008 else static if (hasPeek!T) 1009 { 1010 @trusted ubyte[] peek(ubyte[] buf) const 1011 in 1012 { 1013 assert(buf.length >= this.length, "Given buffer is smaller than the local buffer."); 1014 } 1015 do 1016 { 1017 enum string msg = "Buffer needs to be at least " ~ digestLength!(T).stringof ~ " bytes " ~ 1018 "big, check " ~ typeof(this).stringof ~ ".length!"; 1019 asArray!(digestLength!T)(buf, msg) = _digest.peek(); 1020 return buf[0 .. digestLength!T]; 1021 } 1022 1023 @trusted ubyte[] peek() const 1024 { 1025 enum len = digestLength!T; 1026 auto buf = new ubyte[len]; 1027 asArray!(digestLength!T)(buf) = _digest.peek(); 1028 return buf; 1029 } 1030 } 1031 } 1032 1033 /// 1034 @system unittest 1035 { 1036 import std.digest.md; 1037 //Simple example 1038 auto hash = new WrapperDigest!MD5(); 1039 hash.put(cast(ubyte) 0); 1040 auto result = hash.finish(); 1041 } 1042 1043 /// 1044 @system unittest 1045 { 1046 //using a supplied buffer 1047 import std.digest.md; 1048 ubyte[16] buf; 1049 auto hash = new WrapperDigest!MD5(); 1050 hash.put(cast(ubyte) 0); 1051 auto result = hash.finish(buf[]); 1052 //The result is now in result (and in buf). If you pass a buffer which is bigger than 1053 //necessary, result will have the correct length, but buf will still have it's original 1054 //length 1055 } 1056 1057 @safe unittest 1058 { 1059 // Test peek & length 1060 import std.digest.crc; 1061 auto hash = new WrapperDigest!CRC32(); 1062 assert(hash.length == 4); 1063 hash.put(cast(const(ubyte[]))"The quick brown fox jumps over the lazy dog"); 1064 assert(hash.peek().toHexString() == "39A34F41"); 1065 ubyte[5] buf; 1066 assert(hash.peek(buf).toHexString() == "39A34F41"); 1067 } 1068 1069 /** 1070 * Securely compares two digest representations while protecting against timing 1071 * attacks. Do not use `==` to compare digest representations. 1072 * 1073 * The attack happens as follows: 1074 * 1075 * $(OL 1076 * $(LI An attacker wants to send harmful data to your server, which 1077 * requires a integrity HMAC SHA1 token signed with a secret.) 1078 * $(LI The length of the token is known to be 40 characters long due to its format, 1079 * so the attacker first sends `"0000000000000000000000000000000000000000"`, 1080 * then `"1000000000000000000000000000000000000000"`, and so on.) 1081 * $(LI The given HMAC token is compared with the expected token using the 1082 * `==` string comparison, which returns `false` as soon as the first wrong 1083 * element is found. If a wrong element is found, then a rejection is sent 1084 * back to the sender.) 1085 * $(LI Eventually, the attacker is able to determine the first character in 1086 * the correct token because the sever takes slightly longer to return a 1087 * rejection. This is due to the comparison moving on to second item in 1088 * the two arrays, seeing they are different, and then sending the rejection.) 1089 * $(LI It may seem like too small of a difference in time for the attacker 1090 * to notice, but security researchers have shown that differences as 1091 * small as $(LINK2 http://www.cs.rice.edu/~dwallach/pub/crosby-timing2009.pdf, 1092 * 20µs can be reliably distinguished) even with network inconsistencies.) 1093 * $(LI Repeat the process for each character until the attacker has the whole 1094 * correct token and the server accepts the harmful data. This can be done 1095 * in a week with the attacker pacing the attack to 10 requests per second 1096 * with only one client.) 1097 * ) 1098 * 1099 * This function defends against this attack by always comparing every single 1100 * item in the array if the two arrays are the same length. Therefore, this 1101 * function is always $(BIGOH n) for ranges of the same length. 1102 * 1103 * This attack can also be mitigated via rate limiting and banning IPs which have too 1104 * many rejected requests. However, this does not completely solve the problem, 1105 * as the attacker could be in control of a bot net. To fully defend against 1106 * the timing attack, rate limiting, banning IPs, and using this function 1107 * should be used together. 1108 * 1109 * Params: 1110 * r1 = A digest representation 1111 * r2 = A digest representation 1112 * Returns: 1113 * `true` if both representations are equal, `false` otherwise 1114 * See_Also: 1115 * $(LINK2 https://en.wikipedia.org/wiki/Timing_attack, The Wikipedia article 1116 * on timing attacks). 1117 */ 1118 bool secureEqual(R1, R2)(R1 r1, R2 r2) 1119 if (isInputRange!R1 && isInputRange!R2 && !isInfinite!R1 && !isInfinite!R2 && 1120 (isIntegral!(ElementEncodingType!R1) || isSomeChar!(ElementEncodingType!R1)) && 1121 !is(CommonType!(ElementEncodingType!R1, ElementEncodingType!R2) == void)) 1122 { 1123 static if (hasLength!R1 && hasLength!R2) 1124 if (r1.length != r2.length) 1125 return false; 1126 1127 int result; 1128 1129 static if (isRandomAccessRange!R1 && isRandomAccessRange!R2 && 1130 hasLength!R1 && hasLength!R2) 1131 { 1132 foreach (i; 0 .. r1.length) 1133 result |= r1[i] ^ r2[i]; 1134 } 1135 else static if (hasLength!R1 && hasLength!R2) 1136 { 1137 // Lengths are the same so we can squeeze out a bit of performance 1138 // by not checking if r2 is empty 1139 for (; !r1.empty; r1.popFront(), r2.popFront()) 1140 { 1141 result |= r1.front ^ r2.front; 1142 } 1143 } 1144 else 1145 { 1146 // Generic case, walk both ranges 1147 for (; !r1.empty; r1.popFront(), r2.popFront()) 1148 { 1149 if (r2.empty) return false; 1150 result |= r1.front ^ r2.front; 1151 } 1152 if (!r2.empty) return false; 1153 } 1154 1155 return result == 0; 1156 } 1157 1158 /// 1159 @system pure unittest 1160 { 1161 import std.digest.hmac : hmac; 1162 import std.digest.sha : SHA1; 1163 import std.string : representation; 1164 1165 // a typical HMAC data integrity verification 1166 auto secret = "A7GZIP6TAQA6OHM7KZ42KB9303CEY0MOV5DD6NTV".representation; 1167 auto data = "data".representation; 1168 1169 auto hex1 = data.hmac!SHA1(secret).toHexString; 1170 auto hex2 = data.hmac!SHA1(secret).toHexString; 1171 auto hex3 = "data1".representation.hmac!SHA1(secret).toHexString; 1172 1173 assert( secureEqual(hex1[], hex2[])); 1174 assert(!secureEqual(hex1[], hex3[])); 1175 } 1176 1177 @system pure unittest 1178 { 1179 import std.internal.test.dummyrange : ReferenceInputRange; 1180 import std.range : takeExactly; 1181 import std.string : representation; 1182 import std.utf : byWchar, byDchar; 1183 1184 { 1185 auto hex1 = "02CA3484C375EDD3C0F08D3F50D119E61077".representation; 1186 auto hex2 = "02CA3484C375EDD3C0F08D3F50D119E610779018".representation; 1187 assert(!secureEqual(hex1, hex2)); 1188 } 1189 { 1190 auto hex1 = "02CA3484C375EDD3C0F08D3F50D119E610779018"w.representation; 1191 auto hex2 = "02CA3484C375EDD3C0F08D3F50D119E610779018"d.representation; 1192 assert(secureEqual(hex1, hex2)); 1193 } 1194 { 1195 auto hex1 = "02CA3484C375EDD3C0F08D3F50D119E610779018".byWchar; 1196 auto hex2 = "02CA3484C375EDD3C0F08D3F50D119E610779018".byDchar; 1197 assert(secureEqual(hex1, hex2)); 1198 } 1199 { 1200 auto hex1 = "02CA3484C375EDD3C0F08D3F50D119E61077".byWchar; 1201 auto hex2 = "02CA3484C375EDD3C0F08D3F50D119E610779018".byDchar; 1202 assert(!secureEqual(hex1, hex2)); 1203 } 1204 { 1205 auto hex1 = new ReferenceInputRange!int([0, 1, 2, 3, 4, 5, 6, 7, 8]).takeExactly(9); 1206 auto hex2 = new ReferenceInputRange!int([0, 1, 2, 3, 4, 5, 6, 7, 8]).takeExactly(9); 1207 assert(secureEqual(hex1, hex2)); 1208 } 1209 { 1210 auto hex1 = new ReferenceInputRange!int([0, 1, 2, 3, 4, 5, 6, 7, 8]).takeExactly(9); 1211 auto hex2 = new ReferenceInputRange!int([0, 1, 2, 3, 4, 5, 6, 7, 9]).takeExactly(9); 1212 assert(!secureEqual(hex1, hex2)); 1213 } 1214 } 1215 1216 /** 1217 * Validates a hex string. 1218 * 1219 * Checks whether all characters following an optional "0x" suffix 1220 * are valid hexadecimal digits. 1221 * 1222 * Params: 1223 * hex = hexdecimal encoded byte array 1224 * Returns: 1225 * true = if valid 1226 */ 1227 bool isHexString(String)(String hex) @safe pure nothrow @nogc 1228 if (isSomeString!String) 1229 { 1230 import std.ascii : isHexDigit; 1231 1232 if ((hex.length >= 2) && (hex[0 .. 2] == "0x")) 1233 { 1234 hex = hex[2 .. $]; 1235 } 1236 1237 foreach (digit; hex) 1238 { 1239 if (!digit.isHexDigit) 1240 { 1241 return false; 1242 } 1243 } 1244 1245 return true; 1246 } 1247 1248 /// 1249 @safe unittest 1250 { 1251 assert(isHexString("0x0123456789ABCDEFabcdef")); 1252 assert(isHexString("0123456789ABCDEFabcdef")); 1253 assert(!isHexString("g")); 1254 assert(!isHexString("#")); 1255 } 1256 1257 /** 1258 * Converts a hex text string to a range of bytes. 1259 * 1260 * The input to this function MUST be valid. 1261 * $(REF isHexString, std, digest) can be used to check for this if needed. 1262 * 1263 * Params: 1264 * hex = String representation of a hexdecimal-encoded byte array. 1265 * Returns: 1266 * A forward range of bytes. 1267 */ 1268 auto fromHexStringAsRange(String)(String hex) @safe pure nothrow @nogc 1269 if (isSomeString!String) 1270 { 1271 return HexStringDecoder!String(hex); 1272 } 1273 1274 /// 1275 @safe unittest 1276 { 1277 import std.range.primitives : ElementType, isForwardRange; 1278 import std.traits : ReturnType; 1279 1280 // The decoder implements a forward range. 1281 static assert(isForwardRange!(ReturnType!(fromHexStringAsRange!string))); 1282 static assert(isForwardRange!(ReturnType!(fromHexStringAsRange!wstring))); 1283 static assert(isForwardRange!(ReturnType!(fromHexStringAsRange!dstring))); 1284 1285 // The element type of the range is always `ubyte`. 1286 static assert( 1287 is(ElementType!(ReturnType!(fromHexStringAsRange!string)) == ubyte) 1288 ); 1289 static assert( 1290 is(ElementType!(ReturnType!(fromHexStringAsRange!wstring)) == ubyte) 1291 ); 1292 static assert( 1293 is(ElementType!(ReturnType!(fromHexStringAsRange!dstring)) == ubyte) 1294 ); 1295 } 1296 1297 @safe unittest 1298 { 1299 import std.array : staticArray; 1300 1301 // `staticArray` consumes the range returned by `fromHexStringAsRange`. 1302 assert("0x0000ff".fromHexStringAsRange.staticArray!3 == [0, 0, 0xFF]); 1303 assert("0x0000ff"w.fromHexStringAsRange.staticArray!3 == [0, 0, 0xFF]); 1304 assert("0x0000ff"d.fromHexStringAsRange.staticArray!3 == [0, 0, 0xFF]); 1305 assert("0xff12ff".fromHexStringAsRange.staticArray!1 == [0xFF]); 1306 assert("0x12ff".fromHexStringAsRange.staticArray!2 == [0x12, 255]); 1307 assert( 1308 "0x3AaAA".fromHexStringAsRange.staticArray!4 == [0x3, 0xAA, 0xAA, 0x00] 1309 ); 1310 } 1311 1312 /** 1313 * Converts a hex text string to a range of bytes. 1314 * 1315 * Params: 1316 * hex = String representation of a hexdecimal-encoded byte array. 1317 * Returns: 1318 * An newly allocated array of bytes. 1319 * Throws: 1320 * Exception on invalid input. 1321 * Example: 1322 * --- 1323 * ubyte[] dby = "0xBA".fromHexString; 1324 * --- 1325 * See_Also: 1326 * $(REF fromHexString, std, digest) for a range version of the function. 1327 */ 1328 ubyte[] fromHexString(String)(String hex) @safe pure 1329 if (isSomeString!String) 1330 { 1331 // This function is trivial, yet necessary for consistency. 1332 // It provides a similar API to its `toHexString` counterpart. 1333 1334 if (!hex.isHexString) 1335 { 1336 import std.conv : text; 1337 1338 throw new Exception( 1339 "The provided character sequence `" 1340 ~ hex.text 1341 ~ "` is not a valid hex string." 1342 ); 1343 } 1344 1345 if ((hex.length >= 2) && (hex[0 .. 2] == "0x")) 1346 { 1347 hex = hex[2 .. $]; 1348 } 1349 1350 auto decoder = HexStringDecoder!String(hex); 1351 auto result = new ubyte[](decoder.length); 1352 1353 size_t idx = 0; 1354 foreach (b; decoder) 1355 { 1356 result[idx++] = b; 1357 } 1358 return result; 1359 } 1360 1361 /// 1362 @safe unittest 1363 { 1364 // Single byte 1365 assert("0xff".fromHexString == [255]); 1366 assert("0xff"w.fromHexString == [255]); 1367 assert("0xff"d.fromHexString == [255]); 1368 assert("0xC0".fromHexString == [192]); 1369 assert("0x00".fromHexString == [0]); 1370 1371 // Nothing 1372 assert("".fromHexString == []); 1373 assert(""w.fromHexString == []); 1374 assert(""d.fromHexString == []); 1375 1376 // Nothing but a prefix 1377 assert("0x".fromHexString == []); 1378 assert("0x"w.fromHexString == []); 1379 assert("0x"d.fromHexString == []); 1380 1381 // Half a byte 1382 assert("0x1".fromHexString == [0x01]); 1383 assert("0x1"w.fromHexString == [0x01]); 1384 assert("0x1"d.fromHexString == [0x01]); 1385 1386 // Mixed case is fine. 1387 assert("0xAf".fromHexString == [0xAF]); 1388 assert("0xaF".fromHexString == [0xAF]); 1389 1390 // Multiple bytes 1391 assert("0xfff".fromHexString == [0x0F, 0xFF]); 1392 assert("0x123AaAa".fromHexString == [0x01, 0x23, 0xAA, 0xAA]); 1393 assert("EBBBBF".fromHexString == [0xEB, 0xBB, 0xBF]); 1394 1395 // md5 sum 1396 assert("d41d8cd98f00b204e9800998ecf8427e".fromHexString == [ 1397 0xD4, 0x1D, 0x8C, 0xD9, 0x8F, 0x00, 0xB2, 0x04, 1398 0xE9, 0x80, 0x09, 0x98, 0xEC, 0xF8, 0x42, 0x7E, 1399 ]); 1400 } 1401 1402 /// 1403 @safe unittest 1404 { 1405 // Cycle self-test 1406 const ubyte[] initial = [0x00, 0x12, 0x34, 0xEB]; 1407 assert(initial == initial.toHexString().fromHexString()); 1408 } 1409 1410 private ubyte hexDigitToByte(dchar hexDigit) @safe pure nothrow @nogc 1411 { 1412 static int hexDigitToByteImpl(dchar hexDigit) 1413 { 1414 if (hexDigit >= '0' && hexDigit <= '9') 1415 { 1416 return hexDigit - '0'; 1417 } 1418 else if (hexDigit >= 'A' && hexDigit <= 'F') 1419 { 1420 return hexDigit - 'A' + 10; 1421 } 1422 else if (hexDigit >= 'a' && hexDigit <= 'f') 1423 { 1424 return hexDigit - 'a' + 10; 1425 } 1426 1427 assert(false, "Cannot convert invalid hex digit."); 1428 } 1429 1430 return hexDigitToByteImpl(hexDigit) & 0xFF; 1431 } 1432 1433 @safe unittest 1434 { 1435 assert(hexDigitToByte('0') == 0x0); 1436 assert(hexDigitToByte('9') == 0x9); 1437 assert(hexDigitToByte('a') == 0xA); 1438 assert(hexDigitToByte('b') == 0xB); 1439 assert(hexDigitToByte('A') == 0xA); 1440 assert(hexDigitToByte('C') == 0xC); 1441 } 1442 1443 private struct HexStringDecoder(String) 1444 if (isSomeString!String) 1445 { 1446 String hex; 1447 ubyte front; 1448 bool empty; 1449 1450 this(String hex) 1451 { 1452 if ((hex.length >= 2) && (hex[0 .. 2] == "0x")) 1453 { 1454 hex = hex[2 .. $]; 1455 } 1456 1457 if (hex.length == 0) 1458 { 1459 empty = true; 1460 return; 1461 } 1462 1463 const oddInputLength = (hex.length % 2 == 1); 1464 1465 if (oddInputLength) 1466 { 1467 front = hexDigitToByte(hex[0]); 1468 hex = hex[1 .. $]; 1469 } 1470 else 1471 { 1472 front = cast(ubyte)(hexDigitToByte(hex[0]) << 4 | hexDigitToByte(hex[1])); 1473 hex = hex[2 .. $]; 1474 } 1475 1476 this.hex = hex; 1477 } 1478 1479 void popFront() 1480 { 1481 if (hex.length == 0) 1482 { 1483 empty = true; 1484 return; 1485 } 1486 1487 front = cast(ubyte)(hexDigitToByte(hex[0]) << 4 | hexDigitToByte(hex[1])); 1488 hex = hex[2 .. $]; 1489 } 1490 1491 typeof(this) save() 1492 { 1493 return this; 1494 } 1495 1496 size_t length() const 1497 { 1498 if (this.empty) 1499 { 1500 return 0; 1501 } 1502 1503 // current front + remainder 1504 return 1 + (hex.length >> 1); 1505 } 1506 } 1507 1508 @safe unittest 1509 { 1510 auto decoder = HexStringDecoder!string(""); 1511 assert(decoder.empty); 1512 assert(decoder.length == 0); 1513 1514 decoder = HexStringDecoder!string("0x"); 1515 assert(decoder.empty); 1516 assert(decoder.length == 0); 1517 } 1518 1519 @safe unittest 1520 { 1521 auto decoder = HexStringDecoder!string("0x0077FF"); 1522 assert(!decoder.empty); 1523 assert(decoder.length == 3); 1524 assert(decoder.front == 0x00); 1525 1526 decoder.popFront(); 1527 assert(!decoder.empty); 1528 assert(decoder.length == 2); 1529 assert(decoder.front == 0x77); 1530 1531 decoder.popFront(); 1532 assert(!decoder.empty); 1533 assert(decoder.length == 1); 1534 assert(decoder.front == 0xFF); 1535 1536 decoder.popFront(); 1537 assert(decoder.length == 0); 1538 assert(decoder.empty); 1539 } 1540 1541 @safe unittest 1542 { 1543 auto decoder = HexStringDecoder!string("0x7FF"); 1544 assert(!decoder.empty); 1545 assert(decoder.length == 2); 1546 assert(decoder.front == 0x07); 1547 1548 decoder.popFront(); 1549 assert(!decoder.empty); 1550 assert(decoder.length == 1); 1551 assert(decoder.front == 0xFF); 1552 1553 decoder.popFront(); 1554 assert(decoder.length == 0); 1555 assert(decoder.empty); 1556 }