perf.js 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. var assert = require('assert');
  2. var uuid = require('../');
  3. var log = console.log;
  4. var generators = {
  5. v1: uuid.v1,
  6. v4: uuid.v4
  7. };
  8. var UUID_FORMAT = {
  9. v1: /[0-9a-f]{8}-[0-9a-f]{4}-1[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}/i,
  10. v4: /[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}/i
  11. };
  12. var N = 1e4;
  13. // Get %'age an actual value differs from the ideal value
  14. function divergence(actual, ideal) {
  15. return Math.round(100*100*(actual - ideal)/ideal)/100;
  16. }
  17. function rate(msg, t) {
  18. log(msg + ': ' + (N / (Date.now() - t) * 1e3 | 0) + ' uuids\/second');
  19. }
  20. for (var version in generators) {
  21. var counts = {}, max = 0;
  22. var generator = generators[version];
  23. var format = UUID_FORMAT[version];
  24. log('\nSanity check ' + N + ' ' + version + ' uuids');
  25. for (var i = 0, ok = 0; i < N; i++) {
  26. id = generator();
  27. if (!format.test(id)) {
  28. throw Error(id + ' is not a valid UUID string');
  29. }
  30. if (id != uuid.unparse(uuid.parse(id))) {
  31. assert(fail, id + ' is not a valid id');
  32. }
  33. // Count digits for our randomness check
  34. if (version == 'v4') {
  35. var digits = id.replace(/-/g, '').split('');
  36. for (var j = digits.length-1; j >= 0; j--) {
  37. var c = digits[j];
  38. max = Math.max(max, counts[c] = (counts[c] || 0) + 1);
  39. }
  40. }
  41. }
  42. // Check randomness for v4 UUIDs
  43. if (version == 'v4') {
  44. // Limit that we get worried about randomness. (Purely empirical choice, this!)
  45. var limit = 2*100*Math.sqrt(1/N);
  46. log('\nChecking v4 randomness. Distribution of Hex Digits (% deviation from ideal)');
  47. for (var i = 0; i < 16; i++) {
  48. var c = i.toString(16);
  49. var bar = '', n = counts[c], p = Math.round(n/max*100|0);
  50. // 1-3,5-8, and D-F: 1:16 odds over 30 digits
  51. var ideal = N*30/16;
  52. if (i == 4) {
  53. // 4: 1:1 odds on 1 digit, plus 1:16 odds on 30 digits
  54. ideal = N*(1 + 30/16);
  55. } else if (i >= 8 && i <= 11) {
  56. // 8-B: 1:4 odds on 1 digit, plus 1:16 odds on 30 digits
  57. ideal = N*(1/4 + 30/16);
  58. } else {
  59. // Otherwise: 1:16 odds on 30 digits
  60. ideal = N*30/16;
  61. }
  62. var d = divergence(n, ideal);
  63. // Draw bar using UTF squares (just for grins)
  64. var s = n/max*50 | 0;
  65. while (s--) bar += '=';
  66. assert(Math.abs(d) < limit, c + ' |' + bar + '| ' + counts[c] + ' (' + d + '% < ' + limit + '%)');
  67. }
  68. }
  69. }
  70. // Perf tests
  71. for (var version in generators) {
  72. log('\nPerformance testing ' + version + ' UUIDs');
  73. var generator = generators[version];
  74. var buf = new uuid.BufferClass(16);
  75. for (var i = 0, t = Date.now(); i < N; i++) generator();
  76. rate('uuid.' + version + '()', t);
  77. for (var i = 0, t = Date.now(); i < N; i++) generator('binary');
  78. rate('uuid.' + version + '(\'binary\')', t);
  79. for (var i = 0, t = Date.now(); i < N; i++) generator('binary', buf);
  80. rate('uuid.' + version + '(\'binary\', buffer)', t);
  81. }