max_depth.phpt 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. --TEST--
  2. Bug #78549: Stack overflow due to nested serialized input
  3. --SKIPIF--
  4. <?php
  5. if (getenv('SKIP_PRELOAD')) die('skip Different order of deprecation messages');
  6. ?>
  7. --FILE--
  8. <?php
  9. function create_nested_data($depth, $prefix, $suffix, $inner = 'i:0;') {
  10. return str_repeat($prefix, $depth) . $inner . str_repeat($suffix, $depth);
  11. }
  12. echo "Invalid max_depth:\n";
  13. try {
  14. unserialize('i:0;', ['max_depth' => 'foo']);
  15. } catch (TypeError $exception) {
  16. echo $exception->getMessage() . "\n";
  17. }
  18. try {
  19. unserialize('i:0;', ['max_depth' => -1]);
  20. } catch (ValueError $exception) {
  21. echo $exception->getMessage() . "\n";
  22. }
  23. echo "Array:\n";
  24. var_dump(unserialize(
  25. create_nested_data(128, 'a:1:{i:0;', '}'),
  26. ['max_depth' => 128]
  27. ) !== false);
  28. var_dump(unserialize(
  29. create_nested_data(129, 'a:1:{i:0;', '}'),
  30. ['max_depth' => 128]
  31. ));
  32. echo "Object:\n";
  33. var_dump(unserialize(
  34. create_nested_data(128, 'O:8:"stdClass":1:{i:0;', '}'),
  35. ['max_depth' => 128]
  36. ) !== false);
  37. var_dump(unserialize(
  38. create_nested_data(129, 'O:8:"stdClass":1:{i:0;', '}'),
  39. ['max_depth' => 128]
  40. ));
  41. // Depth can also be adjusted using ini setting
  42. echo "Ini setting:\n";
  43. ini_set("unserialize_max_depth", 128);
  44. var_dump(unserialize(create_nested_data(128, 'a:1:{i:0;', '}')) !== false);
  45. var_dump(unserialize(create_nested_data(129, 'a:1:{i:0;', '}')));
  46. // But an explicitly specified depth still takes precedence
  47. echo "Ini setting overridden:\n";
  48. var_dump(unserialize(
  49. create_nested_data(256, 'a:1:{i:0;', '}'),
  50. ['max_depth' => 256]
  51. ) !== false);
  52. var_dump(unserialize(
  53. create_nested_data(257, 'a:1:{i:0;', '}'),
  54. ['max_depth' => 256]
  55. ));
  56. // Reset ini setting to a large value,
  57. // so it's clear that it won't be used in the following.
  58. ini_set("unserialize_max_depth", 4096);
  59. class Test implements Serializable {
  60. public function serialize() {
  61. return '';
  62. }
  63. public function unserialize($str) {
  64. // Should fail, due to combined nesting level
  65. var_dump(unserialize(create_nested_data(129, 'a:1:{i:0;', '}')));
  66. // Should succeed, below combined nesting level
  67. var_dump(unserialize(create_nested_data(128, 'a:1:{i:0;', '}')) !== false);
  68. }
  69. }
  70. echo "Nested unserialize combined depth limit:\n";
  71. var_dump(is_array(unserialize(
  72. create_nested_data(128, 'a:1:{i:0;', '}', 'C:4:"Test":0:{}'),
  73. ['max_depth' => 256]
  74. )));
  75. class Test2 implements Serializable {
  76. public function serialize() {
  77. return '';
  78. }
  79. public function unserialize($str) {
  80. // If depth limit is overridden, the depth should be counted
  81. // from zero again.
  82. var_dump(unserialize(
  83. create_nested_data(257, 'a:1:{i:0;', '}'),
  84. ['max_depth' => 256]
  85. ));
  86. var_dump(unserialize(
  87. create_nested_data(256, 'a:1:{i:0;', '}'),
  88. ['max_depth' => 256]
  89. ) !== false);
  90. }
  91. }
  92. echo "Nested unserialize overridden depth limit:\n";
  93. var_dump(is_array(unserialize(
  94. create_nested_data(64, 'a:1:{i:0;', '}', 'C:5:"Test2":0:{}'),
  95. ['max_depth' => 128]
  96. )));
  97. ?>
  98. --EXPECTF--
  99. Invalid max_depth:
  100. unserialize(): Option "max_depth" must be of type int, string given
  101. unserialize(): Option "max_depth" must be greater than or equal to 0
  102. Array:
  103. bool(true)
  104. Warning: unserialize(): Maximum depth of 128 exceeded. The depth limit can be changed using the max_depth unserialize() option or the unserialize_max_depth ini setting in %s on line %d
  105. Notice: unserialize(): Error at offset 1157 of 1294 bytes in %s on line %d
  106. bool(false)
  107. Object:
  108. bool(true)
  109. Warning: unserialize(): Maximum depth of 128 exceeded. The depth limit can be changed using the max_depth unserialize() option or the unserialize_max_depth ini setting in %s on line %d
  110. Notice: unserialize(): Error at offset 2834 of 2971 bytes in %s on line %d
  111. bool(false)
  112. Ini setting:
  113. bool(true)
  114. Warning: unserialize(): Maximum depth of 128 exceeded. The depth limit can be changed using the max_depth unserialize() option or the unserialize_max_depth ini setting in %s on line %d
  115. Notice: unserialize(): Error at offset 1157 of 1294 bytes in %s on line %d
  116. bool(false)
  117. Ini setting overridden:
  118. bool(true)
  119. Warning: unserialize(): Maximum depth of 256 exceeded. The depth limit can be changed using the max_depth unserialize() option or the unserialize_max_depth ini setting in %s on line %d
  120. Notice: unserialize(): Error at offset 2309 of 2574 bytes in %s on line %d
  121. bool(false)
  122. Deprecated: %s implements the Serializable interface, which is deprecated. Implement __serialize() and __unserialize() instead (or in addition, if support for old PHP versions is necessary) in %s on line %d
  123. Nested unserialize combined depth limit:
  124. Warning: unserialize(): Maximum depth of 256 exceeded. The depth limit can be changed using the max_depth unserialize() option or the unserialize_max_depth ini setting in %s on line %d
  125. Notice: unserialize(): Error at offset 1157 of 1294 bytes in %s on line %d
  126. bool(false)
  127. bool(true)
  128. bool(true)
  129. Deprecated: %s implements the Serializable interface, which is deprecated. Implement __serialize() and __unserialize() instead (or in addition, if support for old PHP versions is necessary) in %s on line %d
  130. Nested unserialize overridden depth limit:
  131. Warning: unserialize(): Maximum depth of 256 exceeded. The depth limit can be changed using the max_depth unserialize() option or the unserialize_max_depth ini setting in %s on line %d
  132. Notice: unserialize(): Error at offset 2309 of 2574 bytes in %s on line %d
  133. bool(false)
  134. bool(true)
  135. bool(true)