UrgentcareCLI 0.1.0 → 0.1.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (308) hide show
  1. checksums.yaml +4 -4
  2. data/Gemfile +3 -0
  3. data/Gemfile.lock +60 -25
  4. data/Notes +1 -1
  5. data/README.md +5 -5
  6. data/Rakefile +2 -0
  7. data/UrgentCare.gemspec +3 -3
  8. data/background.jpg +0 -0
  9. data/lib/UrgentCare.rb +3 -0
  10. data/lib/UrgentCare/CLI.rb +6 -2
  11. data/lib/UrgentCare/version.rb +1 -1
  12. data/lib/Urgentcare/Scraper.rb +65 -23
  13. data/node_modules/.bin/rimraf +1 -0
  14. data/node_modules/.package-lock.json +250 -0
  15. data/node_modules/balanced-match/.github/FUNDING.yml +2 -0
  16. data/node_modules/balanced-match/LICENSE.md +21 -0
  17. data/node_modules/balanced-match/README.md +97 -0
  18. data/node_modules/balanced-match/index.js +62 -0
  19. data/node_modules/balanced-match/package.json +48 -0
  20. data/node_modules/brace-expansion/LICENSE +21 -0
  21. data/node_modules/brace-expansion/README.md +129 -0
  22. data/node_modules/brace-expansion/index.js +201 -0
  23. data/node_modules/brace-expansion/package.json +47 -0
  24. data/node_modules/concat-map/.travis.yml +4 -0
  25. data/node_modules/concat-map/LICENSE +18 -0
  26. data/node_modules/concat-map/README.markdown +62 -0
  27. data/node_modules/concat-map/example/map.js +6 -0
  28. data/node_modules/concat-map/index.js +13 -0
  29. data/node_modules/concat-map/package.json +43 -0
  30. data/node_modules/concat-map/test/map.js +39 -0
  31. data/node_modules/core-util-is/LICENSE +19 -0
  32. data/node_modules/core-util-is/README.md +3 -0
  33. data/node_modules/core-util-is/float.patch +604 -0
  34. data/node_modules/core-util-is/lib/util.js +107 -0
  35. data/node_modules/core-util-is/package.json +32 -0
  36. data/node_modules/core-util-is/test.js +68 -0
  37. data/node_modules/fs.realpath/LICENSE +43 -0
  38. data/node_modules/fs.realpath/README.md +33 -0
  39. data/node_modules/fs.realpath/index.js +66 -0
  40. data/node_modules/fs.realpath/old.js +303 -0
  41. data/node_modules/fs.realpath/package.json +26 -0
  42. data/node_modules/glob/LICENSE +21 -0
  43. data/node_modules/glob/README.md +375 -0
  44. data/node_modules/glob/changelog.md +67 -0
  45. data/node_modules/glob/common.js +234 -0
  46. data/node_modules/glob/glob.js +788 -0
  47. data/node_modules/glob/package.json +51 -0
  48. data/node_modules/glob/sync.js +484 -0
  49. data/node_modules/immediate/LICENSE.txt +20 -0
  50. data/node_modules/immediate/README.md +93 -0
  51. data/node_modules/immediate/dist/immediate.js +75 -0
  52. data/node_modules/immediate/dist/immediate.min.js +1 -0
  53. data/node_modules/immediate/lib/browser.js +69 -0
  54. data/node_modules/immediate/lib/index.js +73 -0
  55. data/node_modules/immediate/package.json +42 -0
  56. data/node_modules/inflight/LICENSE +15 -0
  57. data/node_modules/inflight/README.md +37 -0
  58. data/node_modules/inflight/inflight.js +54 -0
  59. data/node_modules/inflight/package.json +29 -0
  60. data/node_modules/inherits/LICENSE +16 -0
  61. data/node_modules/inherits/README.md +42 -0
  62. data/node_modules/inherits/inherits.js +9 -0
  63. data/node_modules/inherits/inherits_browser.js +27 -0
  64. data/node_modules/inherits/package.json +29 -0
  65. data/node_modules/isarray/.npmignore +1 -0
  66. data/node_modules/isarray/.travis.yml +4 -0
  67. data/node_modules/isarray/Makefile +6 -0
  68. data/node_modules/isarray/README.md +60 -0
  69. data/node_modules/isarray/component.json +19 -0
  70. data/node_modules/isarray/index.js +5 -0
  71. data/node_modules/isarray/package.json +45 -0
  72. data/node_modules/isarray/test.js +20 -0
  73. data/node_modules/jszip/.codeclimate.yml +16 -0
  74. data/node_modules/jszip/.editorconfig +8 -0
  75. data/node_modules/jszip/.jshintignore +1 -0
  76. data/node_modules/jszip/.jshintrc +21 -0
  77. data/node_modules/jszip/.travis.yml +17 -0
  78. data/node_modules/jszip/CHANGES.md +163 -0
  79. data/node_modules/jszip/LICENSE.markdown +651 -0
  80. data/node_modules/jszip/README.markdown +35 -0
  81. data/node_modules/jszip/dist/jszip.js +30 -0
  82. data/node_modules/jszip/dist/jszip.min.js +13 -0
  83. data/node_modules/jszip/index.d.ts +270 -0
  84. data/node_modules/jszip/lib/base64.js +106 -0
  85. data/node_modules/jszip/lib/compressedObject.js +74 -0
  86. data/node_modules/jszip/lib/compressions.js +14 -0
  87. data/node_modules/jszip/lib/crc32.js +77 -0
  88. data/node_modules/jszip/lib/defaults.js +11 -0
  89. data/node_modules/jszip/lib/external.js +19 -0
  90. data/node_modules/jszip/lib/flate.js +85 -0
  91. data/node_modules/jszip/lib/generate/ZipFileWorker.js +540 -0
  92. data/node_modules/jszip/lib/generate/index.js +57 -0
  93. data/node_modules/jszip/lib/index.js +52 -0
  94. data/node_modules/jszip/lib/license_header.js +11 -0
  95. data/node_modules/jszip/lib/load.js +81 -0
  96. data/node_modules/jszip/lib/nodejs/NodejsStreamInputAdapter.js +74 -0
  97. data/node_modules/jszip/lib/nodejs/NodejsStreamOutputAdapter.js +42 -0
  98. data/node_modules/jszip/lib/nodejsUtils.js +57 -0
  99. data/node_modules/jszip/lib/object.js +389 -0
  100. data/node_modules/jszip/lib/readable-stream-browser.js +9 -0
  101. data/node_modules/jszip/lib/reader/ArrayReader.js +57 -0
  102. data/node_modules/jszip/lib/reader/DataReader.js +116 -0
  103. data/node_modules/jszip/lib/reader/NodeBufferReader.js +19 -0
  104. data/node_modules/jszip/lib/reader/StringReader.js +38 -0
  105. data/node_modules/jszip/lib/reader/Uint8ArrayReader.js +22 -0
  106. data/node_modules/jszip/lib/reader/readerFor.js +28 -0
  107. data/node_modules/jszip/lib/signature.js +7 -0
  108. data/node_modules/jszip/lib/stream/ConvertWorker.js +26 -0
  109. data/node_modules/jszip/lib/stream/Crc32Probe.js +24 -0
  110. data/node_modules/jszip/lib/stream/DataLengthProbe.js +29 -0
  111. data/node_modules/jszip/lib/stream/DataWorker.js +116 -0
  112. data/node_modules/jszip/lib/stream/GenericWorker.js +263 -0
  113. data/node_modules/jszip/lib/stream/StreamHelper.js +212 -0
  114. data/node_modules/jszip/lib/support.js +38 -0
  115. data/node_modules/jszip/lib/utf8.js +275 -0
  116. data/node_modules/jszip/lib/utils.js +476 -0
  117. data/node_modules/jszip/lib/zipEntries.js +262 -0
  118. data/node_modules/jszip/lib/zipEntry.js +294 -0
  119. data/node_modules/jszip/lib/zipObject.js +133 -0
  120. data/node_modules/jszip/package.json +63 -0
  121. data/node_modules/jszip/vendor/FileSaver.js +247 -0
  122. data/node_modules/lie/README.md +62 -0
  123. data/node_modules/lie/dist/lie.js +350 -0
  124. data/node_modules/lie/dist/lie.min.js +1 -0
  125. data/node_modules/lie/dist/lie.polyfill.js +358 -0
  126. data/node_modules/lie/dist/lie.polyfill.min.js +1 -0
  127. data/node_modules/lie/lib/browser.js +273 -0
  128. data/node_modules/lie/lib/index.js +298 -0
  129. data/node_modules/lie/license.md +7 -0
  130. data/node_modules/lie/lie.d.ts +244 -0
  131. data/node_modules/lie/package.json +69 -0
  132. data/node_modules/lie/polyfill.js +4 -0
  133. data/node_modules/minimatch/LICENSE +15 -0
  134. data/node_modules/minimatch/README.md +209 -0
  135. data/node_modules/minimatch/minimatch.js +923 -0
  136. data/node_modules/minimatch/package.json +30 -0
  137. data/node_modules/once/LICENSE +15 -0
  138. data/node_modules/once/README.md +79 -0
  139. data/node_modules/once/once.js +42 -0
  140. data/node_modules/once/package.json +33 -0
  141. data/node_modules/pako/CHANGELOG.md +164 -0
  142. data/node_modules/pako/LICENSE +21 -0
  143. data/node_modules/pako/README.md +191 -0
  144. data/node_modules/pako/dist/pako.js +6818 -0
  145. data/node_modules/pako/dist/pako.min.js +1 -0
  146. data/node_modules/pako/dist/pako_deflate.js +3997 -0
  147. data/node_modules/pako/dist/pako_deflate.min.js +1 -0
  148. data/node_modules/pako/dist/pako_inflate.js +3300 -0
  149. data/node_modules/pako/dist/pako_inflate.min.js +1 -0
  150. data/node_modules/pako/index.js +14 -0
  151. data/node_modules/pako/lib/deflate.js +400 -0
  152. data/node_modules/pako/lib/inflate.js +423 -0
  153. data/node_modules/pako/lib/utils/common.js +105 -0
  154. data/node_modules/pako/lib/utils/strings.js +187 -0
  155. data/node_modules/pako/lib/zlib/README +59 -0
  156. data/node_modules/pako/lib/zlib/adler32.js +51 -0
  157. data/node_modules/pako/lib/zlib/constants.js +68 -0
  158. data/node_modules/pako/lib/zlib/crc32.js +59 -0
  159. data/node_modules/pako/lib/zlib/deflate.js +1874 -0
  160. data/node_modules/pako/lib/zlib/gzheader.js +58 -0
  161. data/node_modules/pako/lib/zlib/inffast.js +345 -0
  162. data/node_modules/pako/lib/zlib/inflate.js +1556 -0
  163. data/node_modules/pako/lib/zlib/inftrees.js +343 -0
  164. data/node_modules/pako/lib/zlib/messages.js +32 -0
  165. data/node_modules/pako/lib/zlib/trees.js +1222 -0
  166. data/node_modules/pako/lib/zlib/zstream.js +47 -0
  167. data/node_modules/pako/package.json +44 -0
  168. data/node_modules/path-is-absolute/index.js +20 -0
  169. data/node_modules/path-is-absolute/license +21 -0
  170. data/node_modules/path-is-absolute/package.json +43 -0
  171. data/node_modules/path-is-absolute/readme.md +59 -0
  172. data/node_modules/process-nextick-args/index.js +45 -0
  173. data/node_modules/process-nextick-args/license.md +19 -0
  174. data/node_modules/process-nextick-args/package.json +25 -0
  175. data/node_modules/process-nextick-args/readme.md +18 -0
  176. data/node_modules/readable-stream/.travis.yml +34 -0
  177. data/node_modules/readable-stream/CONTRIBUTING.md +38 -0
  178. data/node_modules/readable-stream/GOVERNANCE.md +136 -0
  179. data/node_modules/readable-stream/LICENSE +47 -0
  180. data/node_modules/readable-stream/README.md +58 -0
  181. data/node_modules/readable-stream/doc/wg-meetings/2015-01-30.md +60 -0
  182. data/node_modules/readable-stream/duplex-browser.js +1 -0
  183. data/node_modules/readable-stream/duplex.js +1 -0
  184. data/node_modules/readable-stream/lib/_stream_duplex.js +131 -0
  185. data/node_modules/readable-stream/lib/_stream_passthrough.js +47 -0
  186. data/node_modules/readable-stream/lib/_stream_readable.js +1019 -0
  187. data/node_modules/readable-stream/lib/_stream_transform.js +214 -0
  188. data/node_modules/readable-stream/lib/_stream_writable.js +687 -0
  189. data/node_modules/readable-stream/lib/internal/streams/BufferList.js +79 -0
  190. data/node_modules/readable-stream/lib/internal/streams/destroy.js +74 -0
  191. data/node_modules/readable-stream/lib/internal/streams/stream-browser.js +1 -0
  192. data/node_modules/readable-stream/lib/internal/streams/stream.js +1 -0
  193. data/node_modules/readable-stream/package.json +52 -0
  194. data/node_modules/readable-stream/passthrough.js +1 -0
  195. data/node_modules/readable-stream/readable-browser.js +7 -0
  196. data/node_modules/readable-stream/readable.js +19 -0
  197. data/node_modules/readable-stream/transform.js +1 -0
  198. data/node_modules/readable-stream/writable-browser.js +1 -0
  199. data/node_modules/readable-stream/writable.js +8 -0
  200. data/node_modules/rimraf/LICENSE +15 -0
  201. data/node_modules/rimraf/README.md +101 -0
  202. data/node_modules/rimraf/bin.js +50 -0
  203. data/node_modules/rimraf/package.json +29 -0
  204. data/node_modules/rimraf/rimraf.js +372 -0
  205. data/node_modules/safe-buffer/LICENSE +21 -0
  206. data/node_modules/safe-buffer/README.md +584 -0
  207. data/node_modules/safe-buffer/index.d.ts +187 -0
  208. data/node_modules/safe-buffer/index.js +62 -0
  209. data/node_modules/safe-buffer/package.json +37 -0
  210. data/node_modules/selenium-webdriver/CHANGES.md +1114 -0
  211. data/node_modules/selenium-webdriver/LICENSE +202 -0
  212. data/node_modules/selenium-webdriver/NOTICE +2 -0
  213. data/node_modules/selenium-webdriver/README.md +229 -0
  214. data/node_modules/selenium-webdriver/chrome.js +295 -0
  215. data/node_modules/selenium-webdriver/chromium.js +829 -0
  216. data/node_modules/selenium-webdriver/devtools/CDPConnection.js +35 -0
  217. data/node_modules/selenium-webdriver/edge.js +224 -0
  218. data/node_modules/selenium-webdriver/example/chrome_android.js +45 -0
  219. data/node_modules/selenium-webdriver/example/chrome_mobile_emulation.js +46 -0
  220. data/node_modules/selenium-webdriver/example/firefox_channels.js +84 -0
  221. data/node_modules/selenium-webdriver/example/google_search.js +50 -0
  222. data/node_modules/selenium-webdriver/example/google_search_test.js +70 -0
  223. data/node_modules/selenium-webdriver/example/headless.js +63 -0
  224. data/node_modules/selenium-webdriver/example/logging.js +64 -0
  225. data/node_modules/selenium-webdriver/firefox.js +789 -0
  226. data/node_modules/selenium-webdriver/http/index.js +324 -0
  227. data/node_modules/selenium-webdriver/http/util.js +172 -0
  228. data/node_modules/selenium-webdriver/ie.js +503 -0
  229. data/node_modules/selenium-webdriver/index.js +825 -0
  230. data/node_modules/selenium-webdriver/io/exec.js +162 -0
  231. data/node_modules/selenium-webdriver/io/index.js +348 -0
  232. data/node_modules/selenium-webdriver/io/zip.js +223 -0
  233. data/node_modules/selenium-webdriver/lib/atoms/find-elements.js +123 -0
  234. data/node_modules/selenium-webdriver/lib/atoms/get-attribute.js +101 -0
  235. data/node_modules/selenium-webdriver/lib/atoms/is-displayed.js +101 -0
  236. data/node_modules/selenium-webdriver/lib/atoms/mutation-listener.js +55 -0
  237. data/node_modules/selenium-webdriver/lib/by.js +415 -0
  238. data/node_modules/selenium-webdriver/lib/capabilities.js +553 -0
  239. data/node_modules/selenium-webdriver/lib/command.js +206 -0
  240. data/node_modules/selenium-webdriver/lib/error.js +605 -0
  241. data/node_modules/selenium-webdriver/lib/http.js +704 -0
  242. data/node_modules/selenium-webdriver/lib/input.js +946 -0
  243. data/node_modules/selenium-webdriver/lib/logging.js +661 -0
  244. data/node_modules/selenium-webdriver/lib/promise.js +285 -0
  245. data/node_modules/selenium-webdriver/lib/proxy.js +212 -0
  246. data/node_modules/selenium-webdriver/lib/session.js +77 -0
  247. data/node_modules/selenium-webdriver/lib/symbols.js +37 -0
  248. data/node_modules/selenium-webdriver/lib/until.js +429 -0
  249. data/node_modules/selenium-webdriver/lib/webdriver.js +2919 -0
  250. data/node_modules/selenium-webdriver/net/index.js +107 -0
  251. data/node_modules/selenium-webdriver/net/portprober.js +75 -0
  252. data/node_modules/selenium-webdriver/opera.js +406 -0
  253. data/node_modules/selenium-webdriver/package.json +54 -0
  254. data/node_modules/selenium-webdriver/proxy.js +32 -0
  255. data/node_modules/selenium-webdriver/remote/index.js +624 -0
  256. data/node_modules/selenium-webdriver/safari.js +168 -0
  257. data/node_modules/selenium-webdriver/testing/index.js +504 -0
  258. data/node_modules/set-immediate-shim/index.js +7 -0
  259. data/node_modules/set-immediate-shim/package.json +34 -0
  260. data/node_modules/set-immediate-shim/readme.md +31 -0
  261. data/node_modules/string_decoder/.travis.yml +50 -0
  262. data/node_modules/string_decoder/LICENSE +48 -0
  263. data/node_modules/string_decoder/README.md +47 -0
  264. data/node_modules/string_decoder/lib/string_decoder.js +296 -0
  265. data/node_modules/string_decoder/package.json +31 -0
  266. data/node_modules/tmp/CHANGELOG.md +288 -0
  267. data/node_modules/tmp/LICENSE +21 -0
  268. data/node_modules/tmp/README.md +365 -0
  269. data/node_modules/tmp/lib/tmp.js +780 -0
  270. data/node_modules/tmp/node_modules/.bin/rimraf +1 -0
  271. data/node_modules/tmp/node_modules/rimraf/CHANGELOG.md +65 -0
  272. data/node_modules/tmp/node_modules/rimraf/LICENSE +15 -0
  273. data/node_modules/tmp/node_modules/rimraf/README.md +101 -0
  274. data/node_modules/tmp/node_modules/rimraf/bin.js +68 -0
  275. data/node_modules/tmp/node_modules/rimraf/package.json +32 -0
  276. data/node_modules/tmp/node_modules/rimraf/rimraf.js +360 -0
  277. data/node_modules/tmp/package.json +58 -0
  278. data/node_modules/util-deprecate/History.md +16 -0
  279. data/node_modules/util-deprecate/LICENSE +24 -0
  280. data/node_modules/util-deprecate/README.md +53 -0
  281. data/node_modules/util-deprecate/browser.js +67 -0
  282. data/node_modules/util-deprecate/node.js +6 -0
  283. data/node_modules/util-deprecate/package.json +27 -0
  284. data/node_modules/wrappy/LICENSE +15 -0
  285. data/node_modules/wrappy/README.md +36 -0
  286. data/node_modules/wrappy/package.json +29 -0
  287. data/node_modules/wrappy/wrappy.js +33 -0
  288. data/node_modules/ws/LICENSE +21 -0
  289. data/node_modules/ws/README.md +496 -0
  290. data/node_modules/ws/browser.js +8 -0
  291. data/node_modules/ws/index.js +10 -0
  292. data/node_modules/ws/lib/buffer-util.js +129 -0
  293. data/node_modules/ws/lib/constants.js +10 -0
  294. data/node_modules/ws/lib/event-target.js +184 -0
  295. data/node_modules/ws/lib/extension.js +223 -0
  296. data/node_modules/ws/lib/limiter.js +55 -0
  297. data/node_modules/ws/lib/permessage-deflate.js +517 -0
  298. data/node_modules/ws/lib/receiver.js +507 -0
  299. data/node_modules/ws/lib/sender.js +405 -0
  300. data/node_modules/ws/lib/stream.js +165 -0
  301. data/node_modules/ws/lib/validation.js +104 -0
  302. data/node_modules/ws/lib/websocket-server.js +418 -0
  303. data/node_modules/ws/lib/websocket.js +942 -0
  304. data/node_modules/ws/package.json +56 -0
  305. data/package-lock.json +458 -0
  306. data/package.json +5 -0
  307. data/selenium.log +1 -0
  308. metadata +314 -19
@@ -0,0 +1,372 @@
1
+ module.exports = rimraf
2
+ rimraf.sync = rimrafSync
3
+
4
+ var assert = require("assert")
5
+ var path = require("path")
6
+ var fs = require("fs")
7
+ var glob = undefined
8
+ try {
9
+ glob = require("glob")
10
+ } catch (_err) {
11
+ // treat glob as optional.
12
+ }
13
+ var _0666 = parseInt('666', 8)
14
+
15
+ var defaultGlobOpts = {
16
+ nosort: true,
17
+ silent: true
18
+ }
19
+
20
+ // for EMFILE handling
21
+ var timeout = 0
22
+
23
+ var isWindows = (process.platform === "win32")
24
+
25
+ function defaults (options) {
26
+ var methods = [
27
+ 'unlink',
28
+ 'chmod',
29
+ 'stat',
30
+ 'lstat',
31
+ 'rmdir',
32
+ 'readdir'
33
+ ]
34
+ methods.forEach(function(m) {
35
+ options[m] = options[m] || fs[m]
36
+ m = m + 'Sync'
37
+ options[m] = options[m] || fs[m]
38
+ })
39
+
40
+ options.maxBusyTries = options.maxBusyTries || 3
41
+ options.emfileWait = options.emfileWait || 1000
42
+ if (options.glob === false) {
43
+ options.disableGlob = true
44
+ }
45
+ if (options.disableGlob !== true && glob === undefined) {
46
+ throw Error('glob dependency not found, set `options.disableGlob = true` if intentional')
47
+ }
48
+ options.disableGlob = options.disableGlob || false
49
+ options.glob = options.glob || defaultGlobOpts
50
+ }
51
+
52
+ function rimraf (p, options, cb) {
53
+ if (typeof options === 'function') {
54
+ cb = options
55
+ options = {}
56
+ }
57
+
58
+ assert(p, 'rimraf: missing path')
59
+ assert.equal(typeof p, 'string', 'rimraf: path should be a string')
60
+ assert.equal(typeof cb, 'function', 'rimraf: callback function required')
61
+ assert(options, 'rimraf: invalid options argument provided')
62
+ assert.equal(typeof options, 'object', 'rimraf: options should be object')
63
+
64
+ defaults(options)
65
+
66
+ var busyTries = 0
67
+ var errState = null
68
+ var n = 0
69
+
70
+ if (options.disableGlob || !glob.hasMagic(p))
71
+ return afterGlob(null, [p])
72
+
73
+ options.lstat(p, function (er, stat) {
74
+ if (!er)
75
+ return afterGlob(null, [p])
76
+
77
+ glob(p, options.glob, afterGlob)
78
+ })
79
+
80
+ function next (er) {
81
+ errState = errState || er
82
+ if (--n === 0)
83
+ cb(errState)
84
+ }
85
+
86
+ function afterGlob (er, results) {
87
+ if (er)
88
+ return cb(er)
89
+
90
+ n = results.length
91
+ if (n === 0)
92
+ return cb()
93
+
94
+ results.forEach(function (p) {
95
+ rimraf_(p, options, function CB (er) {
96
+ if (er) {
97
+ if ((er.code === "EBUSY" || er.code === "ENOTEMPTY" || er.code === "EPERM") &&
98
+ busyTries < options.maxBusyTries) {
99
+ busyTries ++
100
+ var time = busyTries * 100
101
+ // try again, with the same exact callback as this one.
102
+ return setTimeout(function () {
103
+ rimraf_(p, options, CB)
104
+ }, time)
105
+ }
106
+
107
+ // this one won't happen if graceful-fs is used.
108
+ if (er.code === "EMFILE" && timeout < options.emfileWait) {
109
+ return setTimeout(function () {
110
+ rimraf_(p, options, CB)
111
+ }, timeout ++)
112
+ }
113
+
114
+ // already gone
115
+ if (er.code === "ENOENT") er = null
116
+ }
117
+
118
+ timeout = 0
119
+ next(er)
120
+ })
121
+ })
122
+ }
123
+ }
124
+
125
+ // Two possible strategies.
126
+ // 1. Assume it's a file. unlink it, then do the dir stuff on EPERM or EISDIR
127
+ // 2. Assume it's a directory. readdir, then do the file stuff on ENOTDIR
128
+ //
129
+ // Both result in an extra syscall when you guess wrong. However, there
130
+ // are likely far more normal files in the world than directories. This
131
+ // is based on the assumption that a the average number of files per
132
+ // directory is >= 1.
133
+ //
134
+ // If anyone ever complains about this, then I guess the strategy could
135
+ // be made configurable somehow. But until then, YAGNI.
136
+ function rimraf_ (p, options, cb) {
137
+ assert(p)
138
+ assert(options)
139
+ assert(typeof cb === 'function')
140
+
141
+ // sunos lets the root user unlink directories, which is... weird.
142
+ // so we have to lstat here and make sure it's not a dir.
143
+ options.lstat(p, function (er, st) {
144
+ if (er && er.code === "ENOENT")
145
+ return cb(null)
146
+
147
+ // Windows can EPERM on stat. Life is suffering.
148
+ if (er && er.code === "EPERM" && isWindows)
149
+ fixWinEPERM(p, options, er, cb)
150
+
151
+ if (st && st.isDirectory())
152
+ return rmdir(p, options, er, cb)
153
+
154
+ options.unlink(p, function (er) {
155
+ if (er) {
156
+ if (er.code === "ENOENT")
157
+ return cb(null)
158
+ if (er.code === "EPERM")
159
+ return (isWindows)
160
+ ? fixWinEPERM(p, options, er, cb)
161
+ : rmdir(p, options, er, cb)
162
+ if (er.code === "EISDIR")
163
+ return rmdir(p, options, er, cb)
164
+ }
165
+ return cb(er)
166
+ })
167
+ })
168
+ }
169
+
170
+ function fixWinEPERM (p, options, er, cb) {
171
+ assert(p)
172
+ assert(options)
173
+ assert(typeof cb === 'function')
174
+ if (er)
175
+ assert(er instanceof Error)
176
+
177
+ options.chmod(p, _0666, function (er2) {
178
+ if (er2)
179
+ cb(er2.code === "ENOENT" ? null : er)
180
+ else
181
+ options.stat(p, function(er3, stats) {
182
+ if (er3)
183
+ cb(er3.code === "ENOENT" ? null : er)
184
+ else if (stats.isDirectory())
185
+ rmdir(p, options, er, cb)
186
+ else
187
+ options.unlink(p, cb)
188
+ })
189
+ })
190
+ }
191
+
192
+ function fixWinEPERMSync (p, options, er) {
193
+ assert(p)
194
+ assert(options)
195
+ if (er)
196
+ assert(er instanceof Error)
197
+
198
+ try {
199
+ options.chmodSync(p, _0666)
200
+ } catch (er2) {
201
+ if (er2.code === "ENOENT")
202
+ return
203
+ else
204
+ throw er
205
+ }
206
+
207
+ try {
208
+ var stats = options.statSync(p)
209
+ } catch (er3) {
210
+ if (er3.code === "ENOENT")
211
+ return
212
+ else
213
+ throw er
214
+ }
215
+
216
+ if (stats.isDirectory())
217
+ rmdirSync(p, options, er)
218
+ else
219
+ options.unlinkSync(p)
220
+ }
221
+
222
+ function rmdir (p, options, originalEr, cb) {
223
+ assert(p)
224
+ assert(options)
225
+ if (originalEr)
226
+ assert(originalEr instanceof Error)
227
+ assert(typeof cb === 'function')
228
+
229
+ // try to rmdir first, and only readdir on ENOTEMPTY or EEXIST (SunOS)
230
+ // if we guessed wrong, and it's not a directory, then
231
+ // raise the original error.
232
+ options.rmdir(p, function (er) {
233
+ if (er && (er.code === "ENOTEMPTY" || er.code === "EEXIST" || er.code === "EPERM"))
234
+ rmkids(p, options, cb)
235
+ else if (er && er.code === "ENOTDIR")
236
+ cb(originalEr)
237
+ else
238
+ cb(er)
239
+ })
240
+ }
241
+
242
+ function rmkids(p, options, cb) {
243
+ assert(p)
244
+ assert(options)
245
+ assert(typeof cb === 'function')
246
+
247
+ options.readdir(p, function (er, files) {
248
+ if (er)
249
+ return cb(er)
250
+ var n = files.length
251
+ if (n === 0)
252
+ return options.rmdir(p, cb)
253
+ var errState
254
+ files.forEach(function (f) {
255
+ rimraf(path.join(p, f), options, function (er) {
256
+ if (errState)
257
+ return
258
+ if (er)
259
+ return cb(errState = er)
260
+ if (--n === 0)
261
+ options.rmdir(p, cb)
262
+ })
263
+ })
264
+ })
265
+ }
266
+
267
+ // this looks simpler, and is strictly *faster*, but will
268
+ // tie up the JavaScript thread and fail on excessively
269
+ // deep directory trees.
270
+ function rimrafSync (p, options) {
271
+ options = options || {}
272
+ defaults(options)
273
+
274
+ assert(p, 'rimraf: missing path')
275
+ assert.equal(typeof p, 'string', 'rimraf: path should be a string')
276
+ assert(options, 'rimraf: missing options')
277
+ assert.equal(typeof options, 'object', 'rimraf: options should be object')
278
+
279
+ var results
280
+
281
+ if (options.disableGlob || !glob.hasMagic(p)) {
282
+ results = [p]
283
+ } else {
284
+ try {
285
+ options.lstatSync(p)
286
+ results = [p]
287
+ } catch (er) {
288
+ results = glob.sync(p, options.glob)
289
+ }
290
+ }
291
+
292
+ if (!results.length)
293
+ return
294
+
295
+ for (var i = 0; i < results.length; i++) {
296
+ var p = results[i]
297
+
298
+ try {
299
+ var st = options.lstatSync(p)
300
+ } catch (er) {
301
+ if (er.code === "ENOENT")
302
+ return
303
+
304
+ // Windows can EPERM on stat. Life is suffering.
305
+ if (er.code === "EPERM" && isWindows)
306
+ fixWinEPERMSync(p, options, er)
307
+ }
308
+
309
+ try {
310
+ // sunos lets the root user unlink directories, which is... weird.
311
+ if (st && st.isDirectory())
312
+ rmdirSync(p, options, null)
313
+ else
314
+ options.unlinkSync(p)
315
+ } catch (er) {
316
+ if (er.code === "ENOENT")
317
+ return
318
+ if (er.code === "EPERM")
319
+ return isWindows ? fixWinEPERMSync(p, options, er) : rmdirSync(p, options, er)
320
+ if (er.code !== "EISDIR")
321
+ throw er
322
+
323
+ rmdirSync(p, options, er)
324
+ }
325
+ }
326
+ }
327
+
328
+ function rmdirSync (p, options, originalEr) {
329
+ assert(p)
330
+ assert(options)
331
+ if (originalEr)
332
+ assert(originalEr instanceof Error)
333
+
334
+ try {
335
+ options.rmdirSync(p)
336
+ } catch (er) {
337
+ if (er.code === "ENOENT")
338
+ return
339
+ if (er.code === "ENOTDIR")
340
+ throw originalEr
341
+ if (er.code === "ENOTEMPTY" || er.code === "EEXIST" || er.code === "EPERM")
342
+ rmkidsSync(p, options)
343
+ }
344
+ }
345
+
346
+ function rmkidsSync (p, options) {
347
+ assert(p)
348
+ assert(options)
349
+ options.readdirSync(p).forEach(function (f) {
350
+ rimrafSync(path.join(p, f), options)
351
+ })
352
+
353
+ // We only end up here once we got ENOTEMPTY at least once, and
354
+ // at this point, we are guaranteed to have removed all the kids.
355
+ // So, we know that it won't be ENOENT or ENOTDIR or anything else.
356
+ // try really hard to delete stuff on windows, because it has a
357
+ // PROFOUNDLY annoying habit of not closing handles promptly when
358
+ // files are deleted, resulting in spurious ENOTEMPTY errors.
359
+ var retries = isWindows ? 100 : 1
360
+ var i = 0
361
+ do {
362
+ var threw = true
363
+ try {
364
+ var ret = options.rmdirSync(p, options)
365
+ threw = false
366
+ return ret
367
+ } finally {
368
+ if (++i < retries && threw)
369
+ continue
370
+ }
371
+ } while (true)
372
+ }
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) Feross Aboukhadijeh
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
@@ -0,0 +1,584 @@
1
+ # safe-buffer [![travis][travis-image]][travis-url] [![npm][npm-image]][npm-url] [![downloads][downloads-image]][downloads-url] [![javascript style guide][standard-image]][standard-url]
2
+
3
+ [travis-image]: https://img.shields.io/travis/feross/safe-buffer/master.svg
4
+ [travis-url]: https://travis-ci.org/feross/safe-buffer
5
+ [npm-image]: https://img.shields.io/npm/v/safe-buffer.svg
6
+ [npm-url]: https://npmjs.org/package/safe-buffer
7
+ [downloads-image]: https://img.shields.io/npm/dm/safe-buffer.svg
8
+ [downloads-url]: https://npmjs.org/package/safe-buffer
9
+ [standard-image]: https://img.shields.io/badge/code_style-standard-brightgreen.svg
10
+ [standard-url]: https://standardjs.com
11
+
12
+ #### Safer Node.js Buffer API
13
+
14
+ **Use the new Node.js Buffer APIs (`Buffer.from`, `Buffer.alloc`,
15
+ `Buffer.allocUnsafe`, `Buffer.allocUnsafeSlow`) in all versions of Node.js.**
16
+
17
+ **Uses the built-in implementation when available.**
18
+
19
+ ## install
20
+
21
+ ```
22
+ npm install safe-buffer
23
+ ```
24
+
25
+ ## usage
26
+
27
+ The goal of this package is to provide a safe replacement for the node.js `Buffer`.
28
+
29
+ It's a drop-in replacement for `Buffer`. You can use it by adding one `require` line to
30
+ the top of your node.js modules:
31
+
32
+ ```js
33
+ var Buffer = require('safe-buffer').Buffer
34
+
35
+ // Existing buffer code will continue to work without issues:
36
+
37
+ new Buffer('hey', 'utf8')
38
+ new Buffer([1, 2, 3], 'utf8')
39
+ new Buffer(obj)
40
+ new Buffer(16) // create an uninitialized buffer (potentially unsafe)
41
+
42
+ // But you can use these new explicit APIs to make clear what you want:
43
+
44
+ Buffer.from('hey', 'utf8') // convert from many types to a Buffer
45
+ Buffer.alloc(16) // create a zero-filled buffer (safe)
46
+ Buffer.allocUnsafe(16) // create an uninitialized buffer (potentially unsafe)
47
+ ```
48
+
49
+ ## api
50
+
51
+ ### Class Method: Buffer.from(array)
52
+ <!-- YAML
53
+ added: v3.0.0
54
+ -->
55
+
56
+ * `array` {Array}
57
+
58
+ Allocates a new `Buffer` using an `array` of octets.
59
+
60
+ ```js
61
+ const buf = Buffer.from([0x62,0x75,0x66,0x66,0x65,0x72]);
62
+ // creates a new Buffer containing ASCII bytes
63
+ // ['b','u','f','f','e','r']
64
+ ```
65
+
66
+ A `TypeError` will be thrown if `array` is not an `Array`.
67
+
68
+ ### Class Method: Buffer.from(arrayBuffer[, byteOffset[, length]])
69
+ <!-- YAML
70
+ added: v5.10.0
71
+ -->
72
+
73
+ * `arrayBuffer` {ArrayBuffer} The `.buffer` property of a `TypedArray` or
74
+ a `new ArrayBuffer()`
75
+ * `byteOffset` {Number} Default: `0`
76
+ * `length` {Number} Default: `arrayBuffer.length - byteOffset`
77
+
78
+ When passed a reference to the `.buffer` property of a `TypedArray` instance,
79
+ the newly created `Buffer` will share the same allocated memory as the
80
+ TypedArray.
81
+
82
+ ```js
83
+ const arr = new Uint16Array(2);
84
+ arr[0] = 5000;
85
+ arr[1] = 4000;
86
+
87
+ const buf = Buffer.from(arr.buffer); // shares the memory with arr;
88
+
89
+ console.log(buf);
90
+ // Prints: <Buffer 88 13 a0 0f>
91
+
92
+ // changing the TypedArray changes the Buffer also
93
+ arr[1] = 6000;
94
+
95
+ console.log(buf);
96
+ // Prints: <Buffer 88 13 70 17>
97
+ ```
98
+
99
+ The optional `byteOffset` and `length` arguments specify a memory range within
100
+ the `arrayBuffer` that will be shared by the `Buffer`.
101
+
102
+ ```js
103
+ const ab = new ArrayBuffer(10);
104
+ const buf = Buffer.from(ab, 0, 2);
105
+ console.log(buf.length);
106
+ // Prints: 2
107
+ ```
108
+
109
+ A `TypeError` will be thrown if `arrayBuffer` is not an `ArrayBuffer`.
110
+
111
+ ### Class Method: Buffer.from(buffer)
112
+ <!-- YAML
113
+ added: v3.0.0
114
+ -->
115
+
116
+ * `buffer` {Buffer}
117
+
118
+ Copies the passed `buffer` data onto a new `Buffer` instance.
119
+
120
+ ```js
121
+ const buf1 = Buffer.from('buffer');
122
+ const buf2 = Buffer.from(buf1);
123
+
124
+ buf1[0] = 0x61;
125
+ console.log(buf1.toString());
126
+ // 'auffer'
127
+ console.log(buf2.toString());
128
+ // 'buffer' (copy is not changed)
129
+ ```
130
+
131
+ A `TypeError` will be thrown if `buffer` is not a `Buffer`.
132
+
133
+ ### Class Method: Buffer.from(str[, encoding])
134
+ <!-- YAML
135
+ added: v5.10.0
136
+ -->
137
+
138
+ * `str` {String} String to encode.
139
+ * `encoding` {String} Encoding to use, Default: `'utf8'`
140
+
141
+ Creates a new `Buffer` containing the given JavaScript string `str`. If
142
+ provided, the `encoding` parameter identifies the character encoding.
143
+ If not provided, `encoding` defaults to `'utf8'`.
144
+
145
+ ```js
146
+ const buf1 = Buffer.from('this is a tést');
147
+ console.log(buf1.toString());
148
+ // prints: this is a tést
149
+ console.log(buf1.toString('ascii'));
150
+ // prints: this is a tC)st
151
+
152
+ const buf2 = Buffer.from('7468697320697320612074c3a97374', 'hex');
153
+ console.log(buf2.toString());
154
+ // prints: this is a tést
155
+ ```
156
+
157
+ A `TypeError` will be thrown if `str` is not a string.
158
+
159
+ ### Class Method: Buffer.alloc(size[, fill[, encoding]])
160
+ <!-- YAML
161
+ added: v5.10.0
162
+ -->
163
+
164
+ * `size` {Number}
165
+ * `fill` {Value} Default: `undefined`
166
+ * `encoding` {String} Default: `utf8`
167
+
168
+ Allocates a new `Buffer` of `size` bytes. If `fill` is `undefined`, the
169
+ `Buffer` will be *zero-filled*.
170
+
171
+ ```js
172
+ const buf = Buffer.alloc(5);
173
+ console.log(buf);
174
+ // <Buffer 00 00 00 00 00>
175
+ ```
176
+
177
+ The `size` must be less than or equal to the value of
178
+ `require('buffer').kMaxLength` (on 64-bit architectures, `kMaxLength` is
179
+ `(2^31)-1`). Otherwise, a [`RangeError`][] is thrown. A zero-length Buffer will
180
+ be created if a `size` less than or equal to 0 is specified.
181
+
182
+ If `fill` is specified, the allocated `Buffer` will be initialized by calling
183
+ `buf.fill(fill)`. See [`buf.fill()`][] for more information.
184
+
185
+ ```js
186
+ const buf = Buffer.alloc(5, 'a');
187
+ console.log(buf);
188
+ // <Buffer 61 61 61 61 61>
189
+ ```
190
+
191
+ If both `fill` and `encoding` are specified, the allocated `Buffer` will be
192
+ initialized by calling `buf.fill(fill, encoding)`. For example:
193
+
194
+ ```js
195
+ const buf = Buffer.alloc(11, 'aGVsbG8gd29ybGQ=', 'base64');
196
+ console.log(buf);
197
+ // <Buffer 68 65 6c 6c 6f 20 77 6f 72 6c 64>
198
+ ```
199
+
200
+ Calling `Buffer.alloc(size)` can be significantly slower than the alternative
201
+ `Buffer.allocUnsafe(size)` but ensures that the newly created `Buffer` instance
202
+ contents will *never contain sensitive data*.
203
+
204
+ A `TypeError` will be thrown if `size` is not a number.
205
+
206
+ ### Class Method: Buffer.allocUnsafe(size)
207
+ <!-- YAML
208
+ added: v5.10.0
209
+ -->
210
+
211
+ * `size` {Number}
212
+
213
+ Allocates a new *non-zero-filled* `Buffer` of `size` bytes. The `size` must
214
+ be less than or equal to the value of `require('buffer').kMaxLength` (on 64-bit
215
+ architectures, `kMaxLength` is `(2^31)-1`). Otherwise, a [`RangeError`][] is
216
+ thrown. A zero-length Buffer will be created if a `size` less than or equal to
217
+ 0 is specified.
218
+
219
+ The underlying memory for `Buffer` instances created in this way is *not
220
+ initialized*. The contents of the newly created `Buffer` are unknown and
221
+ *may contain sensitive data*. Use [`buf.fill(0)`][] to initialize such
222
+ `Buffer` instances to zeroes.
223
+
224
+ ```js
225
+ const buf = Buffer.allocUnsafe(5);
226
+ console.log(buf);
227
+ // <Buffer 78 e0 82 02 01>
228
+ // (octets will be different, every time)
229
+ buf.fill(0);
230
+ console.log(buf);
231
+ // <Buffer 00 00 00 00 00>
232
+ ```
233
+
234
+ A `TypeError` will be thrown if `size` is not a number.
235
+
236
+ Note that the `Buffer` module pre-allocates an internal `Buffer` instance of
237
+ size `Buffer.poolSize` that is used as a pool for the fast allocation of new
238
+ `Buffer` instances created using `Buffer.allocUnsafe(size)` (and the deprecated
239
+ `new Buffer(size)` constructor) only when `size` is less than or equal to
240
+ `Buffer.poolSize >> 1` (floor of `Buffer.poolSize` divided by two). The default
241
+ value of `Buffer.poolSize` is `8192` but can be modified.
242
+
243
+ Use of this pre-allocated internal memory pool is a key difference between
244
+ calling `Buffer.alloc(size, fill)` vs. `Buffer.allocUnsafe(size).fill(fill)`.
245
+ Specifically, `Buffer.alloc(size, fill)` will *never* use the internal Buffer
246
+ pool, while `Buffer.allocUnsafe(size).fill(fill)` *will* use the internal
247
+ Buffer pool if `size` is less than or equal to half `Buffer.poolSize`. The
248
+ difference is subtle but can be important when an application requires the
249
+ additional performance that `Buffer.allocUnsafe(size)` provides.
250
+
251
+ ### Class Method: Buffer.allocUnsafeSlow(size)
252
+ <!-- YAML
253
+ added: v5.10.0
254
+ -->
255
+
256
+ * `size` {Number}
257
+
258
+ Allocates a new *non-zero-filled* and non-pooled `Buffer` of `size` bytes. The
259
+ `size` must be less than or equal to the value of
260
+ `require('buffer').kMaxLength` (on 64-bit architectures, `kMaxLength` is
261
+ `(2^31)-1`). Otherwise, a [`RangeError`][] is thrown. A zero-length Buffer will
262
+ be created if a `size` less than or equal to 0 is specified.
263
+
264
+ The underlying memory for `Buffer` instances created in this way is *not
265
+ initialized*. The contents of the newly created `Buffer` are unknown and
266
+ *may contain sensitive data*. Use [`buf.fill(0)`][] to initialize such
267
+ `Buffer` instances to zeroes.
268
+
269
+ When using `Buffer.allocUnsafe()` to allocate new `Buffer` instances,
270
+ allocations under 4KB are, by default, sliced from a single pre-allocated
271
+ `Buffer`. This allows applications to avoid the garbage collection overhead of
272
+ creating many individually allocated Buffers. This approach improves both
273
+ performance and memory usage by eliminating the need to track and cleanup as
274
+ many `Persistent` objects.
275
+
276
+ However, in the case where a developer may need to retain a small chunk of
277
+ memory from a pool for an indeterminate amount of time, it may be appropriate
278
+ to create an un-pooled Buffer instance using `Buffer.allocUnsafeSlow()` then
279
+ copy out the relevant bits.
280
+
281
+ ```js
282
+ // need to keep around a few small chunks of memory
283
+ const store = [];
284
+
285
+ socket.on('readable', () => {
286
+ const data = socket.read();
287
+ // allocate for retained data
288
+ const sb = Buffer.allocUnsafeSlow(10);
289
+ // copy the data into the new allocation
290
+ data.copy(sb, 0, 0, 10);
291
+ store.push(sb);
292
+ });
293
+ ```
294
+
295
+ Use of `Buffer.allocUnsafeSlow()` should be used only as a last resort *after*
296
+ a developer has observed undue memory retention in their applications.
297
+
298
+ A `TypeError` will be thrown if `size` is not a number.
299
+
300
+ ### All the Rest
301
+
302
+ The rest of the `Buffer` API is exactly the same as in node.js.
303
+ [See the docs](https://nodejs.org/api/buffer.html).
304
+
305
+
306
+ ## Related links
307
+
308
+ - [Node.js issue: Buffer(number) is unsafe](https://github.com/nodejs/node/issues/4660)
309
+ - [Node.js Enhancement Proposal: Buffer.from/Buffer.alloc/Buffer.zalloc/Buffer() soft-deprecate](https://github.com/nodejs/node-eps/pull/4)
310
+
311
+ ## Why is `Buffer` unsafe?
312
+
313
+ Today, the node.js `Buffer` constructor is overloaded to handle many different argument
314
+ types like `String`, `Array`, `Object`, `TypedArrayView` (`Uint8Array`, etc.),
315
+ `ArrayBuffer`, and also `Number`.
316
+
317
+ The API is optimized for convenience: you can throw any type at it, and it will try to do
318
+ what you want.
319
+
320
+ Because the Buffer constructor is so powerful, you often see code like this:
321
+
322
+ ```js
323
+ // Convert UTF-8 strings to hex
324
+ function toHex (str) {
325
+ return new Buffer(str).toString('hex')
326
+ }
327
+ ```
328
+
329
+ ***But what happens if `toHex` is called with a `Number` argument?***
330
+
331
+ ### Remote Memory Disclosure
332
+
333
+ If an attacker can make your program call the `Buffer` constructor with a `Number`
334
+ argument, then they can make it allocate uninitialized memory from the node.js process.
335
+ This could potentially disclose TLS private keys, user data, or database passwords.
336
+
337
+ When the `Buffer` constructor is passed a `Number` argument, it returns an
338
+ **UNINITIALIZED** block of memory of the specified `size`. When you create a `Buffer` like
339
+ this, you **MUST** overwrite the contents before returning it to the user.
340
+
341
+ From the [node.js docs](https://nodejs.org/api/buffer.html#buffer_new_buffer_size):
342
+
343
+ > `new Buffer(size)`
344
+ >
345
+ > - `size` Number
346
+ >
347
+ > The underlying memory for `Buffer` instances created in this way is not initialized.
348
+ > **The contents of a newly created `Buffer` are unknown and could contain sensitive
349
+ > data.** Use `buf.fill(0)` to initialize a Buffer to zeroes.
350
+
351
+ (Emphasis our own.)
352
+
353
+ Whenever the programmer intended to create an uninitialized `Buffer` you often see code
354
+ like this:
355
+
356
+ ```js
357
+ var buf = new Buffer(16)
358
+
359
+ // Immediately overwrite the uninitialized buffer with data from another buffer
360
+ for (var i = 0; i < buf.length; i++) {
361
+ buf[i] = otherBuf[i]
362
+ }
363
+ ```
364
+
365
+
366
+ ### Would this ever be a problem in real code?
367
+
368
+ Yes. It's surprisingly common to forget to check the type of your variables in a
369
+ dynamically-typed language like JavaScript.
370
+
371
+ Usually the consequences of assuming the wrong type is that your program crashes with an
372
+ uncaught exception. But the failure mode for forgetting to check the type of arguments to
373
+ the `Buffer` constructor is more catastrophic.
374
+
375
+ Here's an example of a vulnerable service that takes a JSON payload and converts it to
376
+ hex:
377
+
378
+ ```js
379
+ // Take a JSON payload {str: "some string"} and convert it to hex
380
+ var server = http.createServer(function (req, res) {
381
+ var data = ''
382
+ req.setEncoding('utf8')
383
+ req.on('data', function (chunk) {
384
+ data += chunk
385
+ })
386
+ req.on('end', function () {
387
+ var body = JSON.parse(data)
388
+ res.end(new Buffer(body.str).toString('hex'))
389
+ })
390
+ })
391
+
392
+ server.listen(8080)
393
+ ```
394
+
395
+ In this example, an http client just has to send:
396
+
397
+ ```json
398
+ {
399
+ "str": 1000
400
+ }
401
+ ```
402
+
403
+ and it will get back 1,000 bytes of uninitialized memory from the server.
404
+
405
+ This is a very serious bug. It's similar in severity to the
406
+ [the Heartbleed bug](http://heartbleed.com/) that allowed disclosure of OpenSSL process
407
+ memory by remote attackers.
408
+
409
+
410
+ ### Which real-world packages were vulnerable?
411
+
412
+ #### [`bittorrent-dht`](https://www.npmjs.com/package/bittorrent-dht)
413
+
414
+ [Mathias Buus](https://github.com/mafintosh) and I
415
+ ([Feross Aboukhadijeh](http://feross.org/)) found this issue in one of our own packages,
416
+ [`bittorrent-dht`](https://www.npmjs.com/package/bittorrent-dht). The bug would allow
417
+ anyone on the internet to send a series of messages to a user of `bittorrent-dht` and get
418
+ them to reveal 20 bytes at a time of uninitialized memory from the node.js process.
419
+
420
+ Here's
421
+ [the commit](https://github.com/feross/bittorrent-dht/commit/6c7da04025d5633699800a99ec3fbadf70ad35b8)
422
+ that fixed it. We released a new fixed version, created a
423
+ [Node Security Project disclosure](https://nodesecurity.io/advisories/68), and deprecated all
424
+ vulnerable versions on npm so users will get a warning to upgrade to a newer version.
425
+
426
+ #### [`ws`](https://www.npmjs.com/package/ws)
427
+
428
+ That got us wondering if there were other vulnerable packages. Sure enough, within a short
429
+ period of time, we found the same issue in [`ws`](https://www.npmjs.com/package/ws), the
430
+ most popular WebSocket implementation in node.js.
431
+
432
+ If certain APIs were called with `Number` parameters instead of `String` or `Buffer` as
433
+ expected, then uninitialized server memory would be disclosed to the remote peer.
434
+
435
+ These were the vulnerable methods:
436
+
437
+ ```js
438
+ socket.send(number)
439
+ socket.ping(number)
440
+ socket.pong(number)
441
+ ```
442
+
443
+ Here's a vulnerable socket server with some echo functionality:
444
+
445
+ ```js
446
+ server.on('connection', function (socket) {
447
+ socket.on('message', function (message) {
448
+ message = JSON.parse(message)
449
+ if (message.type === 'echo') {
450
+ socket.send(message.data) // send back the user's message
451
+ }
452
+ })
453
+ })
454
+ ```
455
+
456
+ `socket.send(number)` called on the server, will disclose server memory.
457
+
458
+ Here's [the release](https://github.com/websockets/ws/releases/tag/1.0.1) where the issue
459
+ was fixed, with a more detailed explanation. Props to
460
+ [Arnout Kazemier](https://github.com/3rd-Eden) for the quick fix. Here's the
461
+ [Node Security Project disclosure](https://nodesecurity.io/advisories/67).
462
+
463
+
464
+ ### What's the solution?
465
+
466
+ It's important that node.js offers a fast way to get memory otherwise performance-critical
467
+ applications would needlessly get a lot slower.
468
+
469
+ But we need a better way to *signal our intent* as programmers. **When we want
470
+ uninitialized memory, we should request it explicitly.**
471
+
472
+ Sensitive functionality should not be packed into a developer-friendly API that loosely
473
+ accepts many different types. This type of API encourages the lazy practice of passing
474
+ variables in without checking the type very carefully.
475
+
476
+ #### A new API: `Buffer.allocUnsafe(number)`
477
+
478
+ The functionality of creating buffers with uninitialized memory should be part of another
479
+ API. We propose `Buffer.allocUnsafe(number)`. This way, it's not part of an API that
480
+ frequently gets user input of all sorts of different types passed into it.
481
+
482
+ ```js
483
+ var buf = Buffer.allocUnsafe(16) // careful, uninitialized memory!
484
+
485
+ // Immediately overwrite the uninitialized buffer with data from another buffer
486
+ for (var i = 0; i < buf.length; i++) {
487
+ buf[i] = otherBuf[i]
488
+ }
489
+ ```
490
+
491
+
492
+ ### How do we fix node.js core?
493
+
494
+ We sent [a PR to node.js core](https://github.com/nodejs/node/pull/4514) (merged as
495
+ `semver-major`) which defends against one case:
496
+
497
+ ```js
498
+ var str = 16
499
+ new Buffer(str, 'utf8')
500
+ ```
501
+
502
+ In this situation, it's implied that the programmer intended the first argument to be a
503
+ string, since they passed an encoding as a second argument. Today, node.js will allocate
504
+ uninitialized memory in the case of `new Buffer(number, encoding)`, which is probably not
505
+ what the programmer intended.
506
+
507
+ But this is only a partial solution, since if the programmer does `new Buffer(variable)`
508
+ (without an `encoding` parameter) there's no way to know what they intended. If `variable`
509
+ is sometimes a number, then uninitialized memory will sometimes be returned.
510
+
511
+ ### What's the real long-term fix?
512
+
513
+ We could deprecate and remove `new Buffer(number)` and use `Buffer.allocUnsafe(number)` when
514
+ we need uninitialized memory. But that would break 1000s of packages.
515
+
516
+ ~~We believe the best solution is to:~~
517
+
518
+ ~~1. Change `new Buffer(number)` to return safe, zeroed-out memory~~
519
+
520
+ ~~2. Create a new API for creating uninitialized Buffers. We propose: `Buffer.allocUnsafe(number)`~~
521
+
522
+ #### Update
523
+
524
+ We now support adding three new APIs:
525
+
526
+ - `Buffer.from(value)` - convert from any type to a buffer
527
+ - `Buffer.alloc(size)` - create a zero-filled buffer
528
+ - `Buffer.allocUnsafe(size)` - create an uninitialized buffer with given size
529
+
530
+ This solves the core problem that affected `ws` and `bittorrent-dht` which is
531
+ `Buffer(variable)` getting tricked into taking a number argument.
532
+
533
+ This way, existing code continues working and the impact on the npm ecosystem will be
534
+ minimal. Over time, npm maintainers can migrate performance-critical code to use
535
+ `Buffer.allocUnsafe(number)` instead of `new Buffer(number)`.
536
+
537
+
538
+ ### Conclusion
539
+
540
+ We think there's a serious design issue with the `Buffer` API as it exists today. It
541
+ promotes insecure software by putting high-risk functionality into a convenient API
542
+ with friendly "developer ergonomics".
543
+
544
+ This wasn't merely a theoretical exercise because we found the issue in some of the
545
+ most popular npm packages.
546
+
547
+ Fortunately, there's an easy fix that can be applied today. Use `safe-buffer` in place of
548
+ `buffer`.
549
+
550
+ ```js
551
+ var Buffer = require('safe-buffer').Buffer
552
+ ```
553
+
554
+ Eventually, we hope that node.js core can switch to this new, safer behavior. We believe
555
+ the impact on the ecosystem would be minimal since it's not a breaking change.
556
+ Well-maintained, popular packages would be updated to use `Buffer.alloc` quickly, while
557
+ older, insecure packages would magically become safe from this attack vector.
558
+
559
+
560
+ ## links
561
+
562
+ - [Node.js PR: buffer: throw if both length and enc are passed](https://github.com/nodejs/node/pull/4514)
563
+ - [Node Security Project disclosure for `ws`](https://nodesecurity.io/advisories/67)
564
+ - [Node Security Project disclosure for`bittorrent-dht`](https://nodesecurity.io/advisories/68)
565
+
566
+
567
+ ## credit
568
+
569
+ The original issues in `bittorrent-dht`
570
+ ([disclosure](https://nodesecurity.io/advisories/68)) and
571
+ `ws` ([disclosure](https://nodesecurity.io/advisories/67)) were discovered by
572
+ [Mathias Buus](https://github.com/mafintosh) and
573
+ [Feross Aboukhadijeh](http://feross.org/).
574
+
575
+ Thanks to [Adam Baldwin](https://github.com/evilpacket) for helping disclose these issues
576
+ and for his work running the [Node Security Project](https://nodesecurity.io/).
577
+
578
+ Thanks to [John Hiesey](https://github.com/jhiesey) for proofreading this README and
579
+ auditing the code.
580
+
581
+
582
+ ## license
583
+
584
+ MIT. Copyright (C) [Feross Aboukhadijeh](http://feross.org)