@cloudant/couchbackup 2.9.10-SNAPSHOT.113 → 2.9.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (399) hide show
  1. package/.scannerwork/report-task.txt +2 -2
  2. package/.scannerwork/scanner-report/analysis-warnings.pb +2 -2
  3. package/.scannerwork/scanner-report/analysis.log +1 -1
  4. package/.scannerwork/scanner-report/changesets-12.pb +1 -0
  5. package/.scannerwork/scanner-report/changesets-13.pb +1 -0
  6. package/.scannerwork/scanner-report/changesets-2.pb +1 -0
  7. package/.scannerwork/scanner-report/changesets-28.pb +1 -0
  8. package/.scannerwork/scanner-report/changesets-29.pb +1 -0
  9. package/.scannerwork/scanner-report/changesets-3.pb +1 -0
  10. package/.scannerwork/scanner-report/changesets-31.pb +1 -0
  11. package/.scannerwork/scanner-report/changesets-34.pb +1 -0
  12. package/.scannerwork/scanner-report/changesets-35.pb +1 -0
  13. package/.scannerwork/scanner-report/changesets-4.pb +1 -0
  14. package/.scannerwork/scanner-report/changesets-67.pb +1 -0
  15. package/.scannerwork/scanner-report/changesets-68.pb +1 -0
  16. package/.scannerwork/scanner-report/changesets-7.pb +1 -0
  17. package/.scannerwork/scanner-report/changesets-8.pb +1 -0
  18. package/.scannerwork/scanner-report/component-1.pb +2 -2
  19. package/.scannerwork/scanner-report/component-10.pb +1 -1
  20. package/.scannerwork/scanner-report/component-11.pb +1 -1
  21. package/.scannerwork/scanner-report/component-12.pb +1 -0
  22. package/.scannerwork/scanner-report/component-13.pb +1 -1
  23. package/.scannerwork/scanner-report/component-14.pb +1 -1
  24. package/.scannerwork/scanner-report/component-19.pb +1 -1
  25. package/.scannerwork/scanner-report/component-2.pb +1 -0
  26. package/.scannerwork/scanner-report/component-20.pb +1 -1
  27. package/.scannerwork/scanner-report/component-21.pb +1 -1
  28. package/.scannerwork/scanner-report/component-22.pb +1 -1
  29. package/.scannerwork/scanner-report/component-23.pb +1 -1
  30. package/.scannerwork/scanner-report/component-24.pb +1 -1
  31. package/.scannerwork/scanner-report/component-25.pb +1 -1
  32. package/.scannerwork/scanner-report/component-26.pb +1 -1
  33. package/.scannerwork/scanner-report/component-27.pb +1 -1
  34. package/.scannerwork/scanner-report/component-28.pb +1 -0
  35. package/.scannerwork/scanner-report/component-29.pb +1 -0
  36. package/.scannerwork/scanner-report/component-3.pb +1 -0
  37. package/.scannerwork/scanner-report/component-30.pb +1 -1
  38. package/.scannerwork/scanner-report/component-31.pb +1 -0
  39. package/.scannerwork/scanner-report/component-32.pb +1 -1
  40. package/.scannerwork/scanner-report/component-33.pb +1 -1
  41. package/.scannerwork/scanner-report/component-34.pb +1 -0
  42. package/.scannerwork/scanner-report/component-35.pb +1 -0
  43. package/.scannerwork/scanner-report/component-4.pb +1 -0
  44. package/.scannerwork/scanner-report/component-47.pb +1 -1
  45. package/.scannerwork/scanner-report/component-48.pb +1 -1
  46. package/.scannerwork/scanner-report/component-49.pb +1 -1
  47. package/.scannerwork/scanner-report/component-5.pb +1 -1
  48. package/.scannerwork/scanner-report/component-50.pb +1 -1
  49. package/.scannerwork/scanner-report/component-51.pb +1 -1
  50. package/.scannerwork/scanner-report/component-52.pb +1 -1
  51. package/.scannerwork/scanner-report/component-54.pb +1 -1
  52. package/.scannerwork/scanner-report/component-56.pb +1 -1
  53. package/.scannerwork/scanner-report/component-59.pb +1 -1
  54. package/.scannerwork/scanner-report/component-6.pb +1 -1
  55. package/.scannerwork/scanner-report/component-60.pb +1 -1
  56. package/.scannerwork/scanner-report/component-65.pb +1 -0
  57. package/.scannerwork/scanner-report/component-66.pb +1 -0
  58. package/.scannerwork/scanner-report/component-67.pb +1 -1
  59. package/.scannerwork/scanner-report/component-68.pb +1 -1
  60. package/.scannerwork/scanner-report/component-7.pb +1 -0
  61. package/.scannerwork/scanner-report/component-8.pb +1 -0
  62. package/.scannerwork/scanner-report/component-9.pb +1 -1
  63. package/.scannerwork/scanner-report/coverages-10.pb +0 -0
  64. package/.scannerwork/scanner-report/coverages-11.pb +0 -0
  65. package/.scannerwork/scanner-report/coverages-13.pb +0 -0
  66. package/.scannerwork/scanner-report/coverages-14.pb +0 -0
  67. package/.scannerwork/scanner-report/coverages-19.pb +0 -0
  68. package/.scannerwork/scanner-report/coverages-2.pb +0 -0
  69. package/.scannerwork/scanner-report/coverages-20.pb +0 -0
  70. package/.scannerwork/scanner-report/coverages-21.pb +0 -0
  71. package/.scannerwork/scanner-report/coverages-22.pb +0 -0
  72. package/.scannerwork/scanner-report/coverages-23.pb +0 -0
  73. package/.scannerwork/scanner-report/coverages-24.pb +0 -0
  74. package/.scannerwork/scanner-report/coverages-25.pb +0 -0
  75. package/.scannerwork/scanner-report/coverages-26.pb +0 -0
  76. package/.scannerwork/scanner-report/coverages-27.pb +0 -0
  77. package/.scannerwork/scanner-report/coverages-28.pb +0 -0
  78. package/.scannerwork/scanner-report/coverages-29.pb +0 -0
  79. package/.scannerwork/scanner-report/coverages-3.pb +0 -0
  80. package/.scannerwork/scanner-report/coverages-30.pb +0 -0
  81. package/.scannerwork/scanner-report/coverages-32.pb +0 -0
  82. package/.scannerwork/scanner-report/coverages-33.pb +0 -0
  83. package/.scannerwork/scanner-report/coverages-34.pb +0 -0
  84. package/.scannerwork/scanner-report/coverages-4.pb +0 -0
  85. package/.scannerwork/scanner-report/coverages-47.pb +0 -0
  86. package/.scannerwork/scanner-report/coverages-48.pb +0 -0
  87. package/.scannerwork/scanner-report/coverages-49.pb +0 -0
  88. package/.scannerwork/scanner-report/coverages-5.pb +0 -0
  89. package/.scannerwork/scanner-report/coverages-50.pb +0 -0
  90. package/.scannerwork/scanner-report/coverages-51.pb +0 -0
  91. package/.scannerwork/scanner-report/coverages-52.pb +0 -0
  92. package/.scannerwork/scanner-report/coverages-56.pb +0 -0
  93. package/.scannerwork/scanner-report/coverages-59.pb +0 -0
  94. package/.scannerwork/scanner-report/coverages-6.pb +0 -0
  95. package/.scannerwork/scanner-report/coverages-60.pb +0 -0
  96. package/.scannerwork/scanner-report/coverages-67.pb +0 -0
  97. package/.scannerwork/scanner-report/coverages-68.pb +0 -0
  98. package/.scannerwork/scanner-report/coverages-7.pb +0 -0
  99. package/.scannerwork/scanner-report/coverages-8.pb +0 -0
  100. package/.scannerwork/scanner-report/coverages-9.pb +0 -0
  101. package/.scannerwork/scanner-report/duplications-2.pb +2 -0
  102. package/.scannerwork/scanner-report/duplications-21.pb +3 -0
  103. package/.scannerwork/scanner-report/duplications-24.pb +0 -2
  104. package/.scannerwork/scanner-report/duplications-26.pb +0 -2
  105. package/.scannerwork/scanner-report/duplications-27.pb +2 -0
  106. package/.scannerwork/scanner-report/duplications-29.pb +2 -0
  107. package/.scannerwork/scanner-report/duplications-32.pb +3 -2
  108. package/.scannerwork/scanner-report/duplications-33.pb +0 -2
  109. package/.scannerwork/scanner-report/duplications-34.pb +2 -0
  110. package/.scannerwork/scanner-report/duplications-4.pb +2 -0
  111. package/.scannerwork/scanner-report/duplications-49.pb +0 -3
  112. package/.scannerwork/scanner-report/duplications-5.pb +0 -2
  113. package/.scannerwork/scanner-report/duplications-52.pb +3 -0
  114. package/.scannerwork/scanner-report/duplications-56.pb +0 -3
  115. package/.scannerwork/scanner-report/duplications-59.pb +2 -2
  116. package/.scannerwork/scanner-report/duplications-6.pb +0 -2
  117. package/.scannerwork/scanner-report/duplications-60.pb +2 -0
  118. package/.scannerwork/scanner-report/issues-21.pb +4 -0
  119. package/.scannerwork/scanner-report/issues-49.pb +5 -3
  120. package/.scannerwork/scanner-report/measures-10.pb +0 -0
  121. package/.scannerwork/scanner-report/measures-11.pb +0 -0
  122. package/.scannerwork/scanner-report/measures-13.pb +0 -0
  123. package/.scannerwork/scanner-report/measures-14.pb +0 -0
  124. package/.scannerwork/scanner-report/measures-19.pb +0 -0
  125. package/.scannerwork/scanner-report/measures-2.pb +0 -0
  126. package/.scannerwork/scanner-report/measures-20.pb +0 -0
  127. package/.scannerwork/scanner-report/measures-21.pb +0 -0
  128. package/.scannerwork/scanner-report/measures-22.pb +0 -0
  129. package/.scannerwork/scanner-report/measures-23.pb +0 -0
  130. package/.scannerwork/scanner-report/measures-24.pb +0 -0
  131. package/.scannerwork/scanner-report/measures-25.pb +0 -0
  132. package/.scannerwork/scanner-report/measures-26.pb +0 -0
  133. package/.scannerwork/scanner-report/measures-27.pb +0 -0
  134. package/.scannerwork/scanner-report/measures-28.pb +0 -0
  135. package/.scannerwork/scanner-report/measures-29.pb +15 -0
  136. package/.scannerwork/scanner-report/measures-3.pb +16 -0
  137. package/.scannerwork/scanner-report/measures-30.pb +0 -0
  138. package/.scannerwork/scanner-report/measures-32.pb +0 -0
  139. package/.scannerwork/scanner-report/measures-33.pb +0 -0
  140. package/.scannerwork/scanner-report/measures-34.pb +0 -0
  141. package/.scannerwork/scanner-report/measures-35.pb +0 -0
  142. package/.scannerwork/scanner-report/measures-4.pb +0 -0
  143. package/.scannerwork/scanner-report/measures-47.pb +0 -0
  144. package/.scannerwork/scanner-report/measures-48.pb +0 -0
  145. package/.scannerwork/scanner-report/measures-49.pb +0 -0
  146. package/.scannerwork/scanner-report/measures-5.pb +0 -0
  147. package/.scannerwork/scanner-report/measures-50.pb +0 -0
  148. package/.scannerwork/scanner-report/measures-51.pb +0 -0
  149. package/.scannerwork/scanner-report/measures-52.pb +0 -0
  150. package/.scannerwork/scanner-report/measures-54.pb +0 -0
  151. package/.scannerwork/scanner-report/measures-56.pb +0 -0
  152. package/.scannerwork/scanner-report/measures-59.pb +0 -0
  153. package/.scannerwork/scanner-report/measures-6.pb +0 -0
  154. package/.scannerwork/scanner-report/measures-60.pb +0 -0
  155. package/.scannerwork/scanner-report/measures-65.pb +0 -0
  156. package/.scannerwork/scanner-report/measures-66.pb +0 -0
  157. package/.scannerwork/scanner-report/measures-67.pb +0 -0
  158. package/.scannerwork/scanner-report/measures-68.pb +0 -0
  159. package/.scannerwork/scanner-report/measures-7.pb +0 -0
  160. package/.scannerwork/scanner-report/measures-8.pb +0 -0
  161. package/.scannerwork/scanner-report/measures-9.pb +0 -0
  162. package/.scannerwork/scanner-report/metadata.pb +0 -0
  163. package/.scannerwork/scanner-report/source-10.txt +257 -53
  164. package/.scannerwork/scanner-report/source-11.txt +100 -96
  165. package/.scannerwork/scanner-report/source-13.txt +129 -509
  166. package/.scannerwork/scanner-report/source-14.txt +83 -59
  167. package/.scannerwork/scanner-report/source-19.txt +88 -255
  168. package/.scannerwork/scanner-report/source-2.txt +75 -0
  169. package/.scannerwork/scanner-report/source-20.txt +20 -96
  170. package/.scannerwork/scanner-report/source-21.txt +228 -162
  171. package/.scannerwork/scanner-report/source-22.txt +22 -115
  172. package/.scannerwork/scanner-report/source-23.txt +584 -123
  173. package/.scannerwork/scanner-report/source-24.txt +18 -59
  174. package/.scannerwork/scanner-report/source-25.txt +142 -40
  175. package/.scannerwork/scanner-report/source-26.txt +132 -77
  176. package/.scannerwork/scanner-report/source-27.txt +124 -83
  177. package/.scannerwork/scanner-report/{source-62.txt → source-28.txt} +13 -6
  178. package/.scannerwork/scanner-report/source-29.txt +293 -0
  179. package/.scannerwork/scanner-report/source-3.txt +178 -0
  180. package/.scannerwork/scanner-report/source-30.txt +63 -389
  181. package/.scannerwork/scanner-report/source-32.txt +90 -252
  182. package/.scannerwork/scanner-report/source-33.txt +41 -118
  183. package/.scannerwork/scanner-report/source-34.txt +225 -0
  184. package/.scannerwork/scanner-report/source-35.txt +47 -0
  185. package/.scannerwork/scanner-report/source-4.txt +92 -0
  186. package/.scannerwork/scanner-report/source-47.txt +38 -11
  187. package/.scannerwork/scanner-report/source-48.txt +45 -133
  188. package/.scannerwork/scanner-report/source-49.txt +329 -207
  189. package/.scannerwork/scanner-report/source-5.txt +87 -166
  190. package/.scannerwork/scanner-report/source-50.txt +10 -347
  191. package/.scannerwork/scanner-report/source-51.txt +83 -19
  192. package/.scannerwork/scanner-report/source-52.txt +82 -31
  193. package/.scannerwork/scanner-report/source-54.txt +167 -47
  194. package/.scannerwork/scanner-report/source-56.txt +394 -95
  195. package/.scannerwork/scanner-report/source-59.txt +161 -196
  196. package/.scannerwork/scanner-report/source-6.txt +145 -156
  197. package/.scannerwork/scanner-report/source-60.txt +158 -18
  198. package/.scannerwork/scanner-report/source-65.txt +509 -0
  199. package/.scannerwork/scanner-report/source-66.txt +509 -0
  200. package/.scannerwork/scanner-report/source-67.txt +77 -509
  201. package/.scannerwork/scanner-report/source-68.txt +56 -167
  202. package/.scannerwork/scanner-report/source-7.txt +80 -0
  203. package/.scannerwork/scanner-report/source-8.txt +54 -0
  204. package/.scannerwork/scanner-report/source-9.txt +14 -39
  205. package/.scannerwork/scanner-report/symbols-10.pb +354 -30
  206. package/.scannerwork/scanner-report/symbols-11.pb +29 -27
  207. package/.scannerwork/scanner-report/symbols-13.pb +33 -0
  208. package/.scannerwork/scanner-report/symbols-14.pb +41 -45
  209. package/.scannerwork/scanner-report/symbols-19.pb +28 -354
  210. package/.scannerwork/scanner-report/symbols-2.pb +42 -0
  211. package/.scannerwork/scanner-report/symbols-20.pb +16 -58
  212. package/.scannerwork/scanner-report/symbols-21.pb +384 -134
  213. package/.scannerwork/scanner-report/symbols-22.pb +14 -33
  214. package/.scannerwork/scanner-report/symbols-23.pb +1241 -113
  215. package/.scannerwork/scanner-report/symbols-24.pb +9 -41
  216. package/.scannerwork/scanner-report/symbols-25.pb +111 -14
  217. package/.scannerwork/scanner-report/symbols-26.pb +84 -43
  218. package/.scannerwork/scanner-report/symbols-27.pb +102 -42
  219. package/.scannerwork/scanner-report/symbols-28.pb +11 -0
  220. package/.scannerwork/scanner-report/symbols-29.pb +486 -0
  221. package/.scannerwork/scanner-report/symbols-3.pb +144 -0
  222. package/.scannerwork/scanner-report/symbols-30.pb +60 -610
  223. package/.scannerwork/scanner-report/symbols-32.pb +97 -486
  224. package/.scannerwork/scanner-report/symbols-33.pb +36 -102
  225. package/.scannerwork/scanner-report/symbols-34.pb +418 -0
  226. package/.scannerwork/scanner-report/symbols-4.pb +44 -0
  227. package/.scannerwork/scanner-report/symbols-47.pb +43 -14
  228. package/.scannerwork/scanner-report/symbols-48.pb +44 -111
  229. package/.scannerwork/scanner-report/symbols-49.pb +455 -355
  230. package/.scannerwork/scanner-report/symbols-5.pb +59 -180
  231. package/.scannerwork/scanner-report/symbols-50.pb +9 -494
  232. package/.scannerwork/scanner-report/symbols-51.pb +69 -11
  233. package/.scannerwork/scanner-report/symbols-52.pb +74 -33
  234. package/.scannerwork/scanner-report/symbols-56.pb +610 -77
  235. package/.scannerwork/scanner-report/symbols-59.pb +178 -416
  236. package/.scannerwork/scanner-report/symbols-6.pb +120 -147
  237. package/.scannerwork/scanner-report/symbols-60.pb +152 -16
  238. package/.scannerwork/scanner-report/symbols-67.pb +30 -0
  239. package/.scannerwork/scanner-report/symbols-68.pb +19 -0
  240. package/.scannerwork/scanner-report/symbols-7.pb +46 -0
  241. package/.scannerwork/scanner-report/symbols-8.pb +15 -0
  242. package/.scannerwork/scanner-report/symbols-9.pb +13 -18
  243. package/.scannerwork/scanner-report/syntax-highlightings-10.pb +255 -73
  244. package/.scannerwork/scanner-report/syntax-highlightings-11.pb +79 -104
  245. package/.scannerwork/scanner-report/syntax-highlightings-13.pb +118 -3736
  246. package/.scannerwork/scanner-report/syntax-highlightings-14.pb +81 -60
  247. package/.scannerwork/scanner-report/syntax-highlightings-19.pb +103 -250
  248. package/.scannerwork/scanner-report/syntax-highlightings-2.pb +78 -0
  249. package/.scannerwork/scanner-report/syntax-highlightings-20.pb +29 -99
  250. package/.scannerwork/scanner-report/syntax-highlightings-21.pb +269 -183
  251. package/.scannerwork/scanner-report/syntax-highlightings-22.pb +22 -97
  252. package/.scannerwork/scanner-report/syntax-highlightings-23.pb +815 -141
  253. package/.scannerwork/scanner-report/syntax-highlightings-24.pb +17 -45
  254. package/.scannerwork/scanner-report/syntax-highlightings-25.pb +193 -24
  255. package/.scannerwork/scanner-report/syntax-highlightings-26.pb +157 -56
  256. package/.scannerwork/scanner-report/syntax-highlightings-27.pb +195 -84
  257. package/.scannerwork/scanner-report/syntax-highlightings-28.pb +59 -0
  258. package/.scannerwork/scanner-report/syntax-highlightings-29.pb +659 -0
  259. package/.scannerwork/scanner-report/syntax-highlightings-3.pb +243 -0
  260. package/.scannerwork/scanner-report/syntax-highlightings-30.pb +74 -612
  261. package/.scannerwork/scanner-report/syntax-highlightings-32.pb +147 -624
  262. package/.scannerwork/scanner-report/syntax-highlightings-33.pb +57 -189
  263. package/.scannerwork/scanner-report/syntax-highlightings-34.pb +331 -0
  264. package/.scannerwork/scanner-report/syntax-highlightings-35.pb +68 -0
  265. package/.scannerwork/scanner-report/syntax-highlightings-4.pb +90 -0
  266. package/.scannerwork/scanner-report/syntax-highlightings-47.pb +47 -19
  267. package/.scannerwork/scanner-report/syntax-highlightings-48.pb +63 -176
  268. package/.scannerwork/scanner-report/syntax-highlightings-49.pb +498 -197
  269. package/.scannerwork/scanner-report/syntax-highlightings-5.pb +100 -192
  270. package/.scannerwork/scanner-report/syntax-highlightings-50.pb +15 -597
  271. package/.scannerwork/scanner-report/syntax-highlightings-51.pb +71 -25
  272. package/.scannerwork/scanner-report/syntax-highlightings-52.pb +118 -55
  273. package/.scannerwork/scanner-report/syntax-highlightings-54.pb +938 -57
  274. package/.scannerwork/scanner-report/syntax-highlightings-56.pb +619 -129
  275. package/.scannerwork/scanner-report/syntax-highlightings-59.pb +170 -276
  276. package/.scannerwork/scanner-report/syntax-highlightings-6.pb +155 -156
  277. package/.scannerwork/scanner-report/syntax-highlightings-60.pb +185 -34
  278. package/.scannerwork/scanner-report/syntax-highlightings-65.pb +3748 -0
  279. package/.scannerwork/scanner-report/syntax-highlightings-66.pb +3747 -0
  280. package/.scannerwork/scanner-report/syntax-highlightings-67.pb +72 -3717
  281. package/.scannerwork/scanner-report/syntax-highlightings-68.pb +54 -921
  282. package/.scannerwork/scanner-report/syntax-highlightings-7.pb +95 -0
  283. package/.scannerwork/scanner-report/syntax-highlightings-8.pb +54 -0
  284. package/.scannerwork/scanner-report/syntax-highlightings-9.pb +33 -70
  285. package/CHANGES.md +6 -2
  286. package/package.json +1 -1
  287. package/test-16-results.xml +150 -150
  288. package/test-18-results.xml +149 -149
  289. package/test-iam-18-results.xml +50 -50
  290. package/.scannerwork/scanner-report/changesets-16.pb +0 -1
  291. package/.scannerwork/scanner-report/changesets-17.pb +0 -1
  292. package/.scannerwork/scanner-report/changesets-18.pb +0 -1
  293. package/.scannerwork/scanner-report/changesets-45.pb +0 -1
  294. package/.scannerwork/scanner-report/changesets-46.pb +0 -1
  295. package/.scannerwork/scanner-report/changesets-53.pb +0 -1
  296. package/.scannerwork/scanner-report/changesets-54.pb +0 -1
  297. package/.scannerwork/scanner-report/changesets-55.pb +0 -1
  298. package/.scannerwork/scanner-report/changesets-57.pb +0 -1
  299. package/.scannerwork/scanner-report/changesets-58.pb +0 -1
  300. package/.scannerwork/scanner-report/changesets-61.pb +0 -1
  301. package/.scannerwork/scanner-report/changesets-62.pb +0 -1
  302. package/.scannerwork/scanner-report/changesets-63.pb +0 -1
  303. package/.scannerwork/scanner-report/changesets-64.pb +0 -1
  304. package/.scannerwork/scanner-report/component-16.pb +0 -1
  305. package/.scannerwork/scanner-report/component-17.pb +0 -1
  306. package/.scannerwork/scanner-report/component-18.pb +0 -1
  307. package/.scannerwork/scanner-report/component-45.pb +0 -1
  308. package/.scannerwork/scanner-report/component-46.pb +0 -1
  309. package/.scannerwork/scanner-report/component-53.pb +0 -1
  310. package/.scannerwork/scanner-report/component-55.pb +0 -1
  311. package/.scannerwork/scanner-report/component-57.pb +0 -1
  312. package/.scannerwork/scanner-report/component-58.pb +0 -1
  313. package/.scannerwork/scanner-report/component-61.pb +0 -1
  314. package/.scannerwork/scanner-report/component-62.pb +0 -1
  315. package/.scannerwork/scanner-report/component-63.pb +0 -1
  316. package/.scannerwork/scanner-report/component-64.pb +0 -1
  317. package/.scannerwork/scanner-report/coverages-16.pb +0 -0
  318. package/.scannerwork/scanner-report/coverages-18.pb +0 -0
  319. package/.scannerwork/scanner-report/coverages-45.pb +0 -0
  320. package/.scannerwork/scanner-report/coverages-53.pb +0 -0
  321. package/.scannerwork/scanner-report/coverages-55.pb +0 -0
  322. package/.scannerwork/scanner-report/coverages-57.pb +0 -0
  323. package/.scannerwork/scanner-report/coverages-58.pb +0 -0
  324. package/.scannerwork/scanner-report/coverages-61.pb +0 -0
  325. package/.scannerwork/scanner-report/coverages-62.pb +0 -0
  326. package/.scannerwork/scanner-report/coverages-63.pb +0 -0
  327. package/.scannerwork/scanner-report/coverages-64.pb +0 -0
  328. package/.scannerwork/scanner-report/duplications-58.pb +0 -0
  329. package/.scannerwork/scanner-report/duplications-61.pb +0 -3
  330. package/.scannerwork/scanner-report/duplications-63.pb +0 -0
  331. package/.scannerwork/scanner-report/duplications-64.pb +0 -0
  332. package/.scannerwork/scanner-report/issues-50.pb +0 -6
  333. package/.scannerwork/scanner-report/measures-16.pb +0 -0
  334. package/.scannerwork/scanner-report/measures-18.pb +0 -0
  335. package/.scannerwork/scanner-report/measures-45.pb +0 -0
  336. package/.scannerwork/scanner-report/measures-53.pb +0 -0
  337. package/.scannerwork/scanner-report/measures-55.pb +0 -0
  338. package/.scannerwork/scanner-report/measures-57.pb +0 -0
  339. package/.scannerwork/scanner-report/measures-58.pb +0 -0
  340. package/.scannerwork/scanner-report/measures-61.pb +0 -0
  341. package/.scannerwork/scanner-report/measures-62.pb +0 -0
  342. package/.scannerwork/scanner-report/measures-63.pb +0 -0
  343. package/.scannerwork/scanner-report/measures-64.pb +0 -0
  344. package/.scannerwork/scanner-report/source-16.txt +0 -118
  345. package/.scannerwork/scanner-report/source-18.txt +0 -31
  346. package/.scannerwork/scanner-report/source-45.txt +0 -625
  347. package/.scannerwork/scanner-report/source-53.txt +0 -92
  348. package/.scannerwork/scanner-report/source-55.txt +0 -147
  349. package/.scannerwork/scanner-report/source-57.txt +0 -68
  350. package/.scannerwork/scanner-report/source-58.txt +0 -63
  351. package/.scannerwork/scanner-report/source-61.txt +0 -131
  352. package/.scannerwork/scanner-report/source-63.txt +0 -34
  353. package/.scannerwork/scanner-report/source-64.txt +0 -100
  354. package/.scannerwork/scanner-report/symbols-16.pb +0 -30
  355. package/.scannerwork/scanner-report/symbols-18.pb +0 -14
  356. package/.scannerwork/scanner-report/symbols-45.pb +0 -1254
  357. package/.scannerwork/scanner-report/symbols-53.pb +0 -60
  358. package/.scannerwork/scanner-report/symbols-55.pb +0 -85
  359. package/.scannerwork/scanner-report/symbols-57.pb +0 -45
  360. package/.scannerwork/scanner-report/symbols-58.pb +0 -43
  361. package/.scannerwork/scanner-report/symbols-61.pb +0 -97
  362. package/.scannerwork/scanner-report/symbols-62.pb +0 -9
  363. package/.scannerwork/scanner-report/symbols-63.pb +0 -10
  364. package/.scannerwork/scanner-report/symbols-64.pb +0 -69
  365. package/.scannerwork/scanner-report/syntax-highlightings-16.pb +0 -113
  366. package/.scannerwork/scanner-report/syntax-highlightings-18.pb +0 -45
  367. package/.scannerwork/scanner-report/syntax-highlightings-45.pb +0 -887
  368. package/.scannerwork/scanner-report/syntax-highlightings-53.pb +0 -110
  369. package/.scannerwork/scanner-report/syntax-highlightings-55.pb +0 -191
  370. package/.scannerwork/scanner-report/syntax-highlightings-57.pb +0 -110
  371. package/.scannerwork/scanner-report/syntax-highlightings-58.pb +0 -82
  372. package/.scannerwork/scanner-report/syntax-highlightings-61.pb +0 -182
  373. package/.scannerwork/scanner-report/syntax-highlightings-62.pb +0 -48
  374. package/.scannerwork/scanner-report/syntax-highlightings-63.pb +0 -50
  375. package/.scannerwork/scanner-report/syntax-highlightings-64.pb +0 -105
  376. /package/.scannerwork/scanner-report/{coverages-17.pb → coverages-12.pb} +0 -0
  377. /package/.scannerwork/scanner-report/{coverages-46.pb → coverages-31.pb} +0 -0
  378. /package/.scannerwork/scanner-report/{coverages-54.pb → coverages-35.pb} +0 -0
  379. /package/.scannerwork/scanner-report/{duplications-16.pb → duplications-12.pb} +0 -0
  380. /package/.scannerwork/scanner-report/{duplications-17.pb → duplications-13.pb} +0 -0
  381. /package/.scannerwork/scanner-report/{duplications-18.pb → duplications-28.pb} +0 -0
  382. /package/.scannerwork/scanner-report/{duplications-45.pb → duplications-3.pb} +0 -0
  383. /package/.scannerwork/scanner-report/{duplications-46.pb → duplications-31.pb} +0 -0
  384. /package/.scannerwork/scanner-report/{duplications-50.pb → duplications-35.pb} +0 -0
  385. /package/.scannerwork/scanner-report/{duplications-53.pb → duplications-67.pb} +0 -0
  386. /package/.scannerwork/scanner-report/{duplications-54.pb → duplications-68.pb} +0 -0
  387. /package/.scannerwork/scanner-report/{duplications-55.pb → duplications-7.pb} +0 -0
  388. /package/.scannerwork/scanner-report/{duplications-57.pb → duplications-8.pb} +0 -0
  389. /package/.scannerwork/scanner-report/{issues-46.pb → issues-31.pb} +0 -0
  390. /package/.scannerwork/scanner-report/{issues-54.pb → issues-35.pb} +0 -0
  391. /package/.scannerwork/scanner-report/{measures-17.pb → measures-12.pb} +0 -0
  392. /package/.scannerwork/scanner-report/{measures-46.pb → measures-31.pb} +0 -0
  393. /package/.scannerwork/scanner-report/{source-17.txt → source-12.txt} +0 -0
  394. /package/.scannerwork/scanner-report/{source-46.txt → source-31.txt} +0 -0
  395. /package/.scannerwork/scanner-report/{symbols-17.pb → symbols-12.pb} +0 -0
  396. /package/.scannerwork/scanner-report/{symbols-46.pb → symbols-31.pb} +0 -0
  397. /package/.scannerwork/scanner-report/{symbols-54.pb → symbols-35.pb} +0 -0
  398. /package/.scannerwork/scanner-report/{syntax-highlightings-17.pb → syntax-highlightings-12.pb} +0 -0
  399. /package/.scannerwork/scanner-report/{syntax-highlightings-46.pb → syntax-highlightings-31.pb} +0 -0
@@ -1,4 +1,4 @@
1
- // Copyright © 2017, 2021 IBM Corp. All rights reserved.
1
+ // Copyright © 2017, 2023 IBM Corp. All rights reserved.
2
2
  //
3
3
  // Licensed under the Apache License, Version 2.0 (the "License");
4
4
  // you may not use this file except in compliance with the License.
@@ -11,154 +11,615 @@
11
11
  // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
12
  // See the License for the specific language governing permissions and
13
13
  // limitations under the License.
14
+
15
+ /* global */
14
16
  'use strict';
15
17
 
16
- const async = require('async');
18
+ const assert = require('assert');
19
+ const spawn = require('child_process').spawn;
20
+ const app = require('../app.js');
21
+ const dbUrl = require('../includes/cliutils.js').databaseUrl;
17
22
  const stream = require('stream');
18
- const error = require('./error.js');
19
- const debug = require('debug')('couchbackup:writer');
20
-
21
- module.exports = function(db, bufferSize, parallelism, ee) {
22
- const writer = new stream.Transform({ objectMode: true });
23
- let buffer = [];
24
- let written = 0;
25
- let linenumber = 0;
26
-
27
- // this is the queue of chunks that are written to the database
28
- // the queue's payload will be an array of documents to be written,
29
- // the size of the array will be bufferSize. The variable parallelism
30
- // determines how many HTTP requests will occur at any one time.
31
- const q = async.queue(function(payload, cb) {
32
- // if we are restoring known revisions, we need to supply new_edits=false
33
- if (payload.docs && payload.docs[0] && payload.docs[0]._rev) {
34
- payload.new_edits = false;
35
- debug('Using new_edits false mode.');
36
- }
37
-
38
- if (!didError) {
39
- db.service.postBulkDocs({
40
- db: db.db,
41
- bulkDocs: payload
42
- }).then(response => {
43
- if (!response.result || (payload.new_edits === false && response.result.length > 0)) {
44
- throw new Error(`Error writing batch with new_edits:${payload.new_edits !== false}` +
45
- ` and ${response.result ? response.result.length : 'unavailable'} items`);
46
- }
47
- written += payload.docs.length;
48
- writer.emit('restored', { documents: payload.docs.length, total: written });
49
- cb();
50
- }).catch(err => {
51
- err = error.convertResponseError(err);
52
- debug(`Error writing docs ${err.name} ${err.message}`);
53
- cb(err, payload);
23
+ const fs = require('fs');
24
+ const zlib = require('zlib');
25
+ const Tail = require('tail').Tail;
26
+ const compare = require('./compare.js');
27
+ const request = require('../includes/request.js');
28
+
29
+ function scenario(test, params) {
30
+ return `${test} ${(params.useApi) ? 'using API' : 'using CLI'}`;
31
+ }
32
+
33
+ function params() {
34
+ const p = {};
35
+ for (let i = 0; i < arguments.length; i++) {
36
+ Object.assign(p, arguments[i]);
37
+ }
38
+ return p;
39
+ }
40
+
41
+ // Returns the event emitter for API calls, or the child process for CLI calls
42
+ function testBackup(params, databaseName, outputStream, callback) {
43
+ let gzip;
44
+ let openssl;
45
+ let backup;
46
+ let backupStream = outputStream;
47
+
48
+ // Configure API key if needed
49
+ augmentParamsWithApiKey(params);
50
+
51
+ // Pipe via compression if requested
52
+ if (params.compression) {
53
+ if (params.useApi) {
54
+ // If use API use the Node zlib stream
55
+ const zlib = require('zlib');
56
+ backupStream = zlib.createGzip();
57
+ backupStream.pipe(outputStream);
58
+ } else {
59
+ // Spawn process for gzip
60
+ gzip = spawn('gzip', [], { stdio: ['pipe', 'pipe', 'inherit'] });
61
+ // Pipe the streams as needed
62
+ gzip.stdout.pipe(outputStream);
63
+ backupStream = gzip.stdin;
64
+ // register an error handler
65
+ gzip.on('error', function(err) {
66
+ callback(err);
54
67
  });
55
68
  }
56
- }, parallelism);
69
+ }
70
+
71
+ // Pipe via encryption if requested
72
+ if (params.encryption) {
73
+ if (params.useApi) {
74
+ // Currently only CLI support for testing encryption
75
+ callback(new Error('Not implemented: cannot test encrypted API backups at this time.'));
76
+ } else {
77
+ // Spawn process for openssl
78
+ openssl = spawn('openssl', ['aes-128-cbc', '-pass', 'pass:12345'], { stdio: ['pipe', 'pipe', 'inherit'] });
79
+ // Pipe the streams as needed
80
+ openssl.stdout.pipe(outputStream);
81
+ backupStream = openssl.stdin;
82
+ // register an error handler
83
+ openssl.on('error', function(err) {
84
+ callback(err);
85
+ });
86
+ }
87
+ }
57
88
 
58
- let didError = false;
89
+ let tail;
90
+ if (params.abort) {
91
+ // Create the log file for abort tests so we can tail it, other tests assert
92
+ // the log file is usually created normally by the backup process.
93
+ const f = fs.openSync(params.opts.log, 'w');
94
+ fs.closeSync(f);
59
95
 
60
- // write the contents of the buffer to CouchDB in blocks of bufferSize
61
- function processBuffer(flush, callback) {
62
- function taskCallback(err, payload) {
63
- if (err && !didError) {
64
- debug(`Queue task failed with error ${err.name}`);
65
- didError = true;
66
- q.kill();
67
- writer.emit('error', err);
96
+ // Use tail to watch the log file for a batch to be completed then abort
97
+ tail = new Tail(params.opts.log, { useWatchFile: true, fsWatchOptions: { interval: 500 }, follow: false });
98
+ tail.on('line', function(data) {
99
+ const matches = data.match(/:d batch\d+/);
100
+ if (matches !== null) {
101
+ // Turn off the tail.
102
+ tail.unwatch();
103
+ // Abort the backup
104
+ backupAbort(params.useApi, backup);
68
105
  }
69
- }
106
+ });
107
+ tail.on('error', function(err) {
108
+ callback(err);
109
+ });
110
+ }
70
111
 
71
- if (flush || buffer.length >= bufferSize) {
72
- // work through the buffer to break off bufferSize chunks
73
- // and feed the chunks to the queue
74
- do {
75
- // split the buffer into bufferSize chunks
76
- const toSend = buffer.splice(0, bufferSize);
112
+ if (params.useApi) {
113
+ backup = app.backup(dbUrl(process.env.COUCH_URL, databaseName), backupStream, params.opts, function(err, data) {
114
+ if (err) {
115
+ if (params.expectedBackupError) {
116
+ try {
117
+ assert.strictEqual(err.name, params.expectedBackupError.name, 'The backup should receive the expected error.');
118
+ // Got the expected error, so wipe it for the callback
119
+ err = null;
120
+ } catch (caught) {
121
+ // Update the error with the assertion failure
122
+ err = caught;
123
+ }
124
+ }
125
+ } else {
126
+ console.log(data);
127
+ }
128
+ callback(err);
129
+ });
130
+ if (backup) {
131
+ backup.on('error', function(err) {
132
+ console.error(`Caught non-fatal error: ${err}`);
133
+ });
134
+ }
135
+ } else {
136
+ // Default to pipe, but will use 'inherit' if using --output (see params.opts.output)
137
+ let destination = 'pipe';
77
138
 
78
- // and add the chunk to the queue
79
- debug(`Adding ${toSend.length} to the write queue.`);
80
- q.push({ docs: toSend }, taskCallback);
81
- } while (buffer.length >= bufferSize);
139
+ // Set up default args
140
+ const args = ['./bin/couchbackup.bin.js', '--db', databaseName];
141
+ if (params.opts) {
142
+ if (params.opts.mode) {
143
+ args.push('--mode');
144
+ args.push(params.opts.mode);
145
+ }
146
+ if (params.opts.output) {
147
+ args.push('--output');
148
+ args.push(params.opts.output);
149
+ destination = 'inherit';
150
+ }
151
+ if (params.opts.log) {
152
+ args.push('--log');
153
+ args.push(params.opts.log);
154
+ }
155
+ if (params.opts.resume) {
156
+ args.push('--resume');
157
+ args.push(params.opts.resume);
158
+ }
159
+ if (params.opts.bufferSize) {
160
+ args.push('--buffer-size');
161
+ args.push(params.opts.bufferSize);
162
+ }
163
+ if (params.opts.iamApiKey) {
164
+ args.push('--iam-api-key');
165
+ args.push(params.opts.iamApiKey);
166
+ }
167
+ }
82
168
 
83
- // send any leftover documents to the queue
84
- if (flush && buffer.length > 0) {
85
- debug(`Adding remaining ${buffer.length} to the write queue.`);
86
- q.push({ docs: buffer }, taskCallback);
169
+ let count = 0;
170
+ /**
171
+ * In some tests we need to wait for both the backup process
172
+ * and the outputStream to "close". If we callback from either
173
+ * event the other might not be ready and lead to flaky tests.
174
+ *
175
+ * This function delegates to the callback but only after the
176
+ * correct number of invocations. That is 2 when we have an
177
+ * output stream or 1 otherwise, and only once in the case of
178
+ * an error.
179
+ */
180
+ function gatingCallback(err) {
181
+ count += 1;
182
+ if (err) {
183
+ if (count === 1) {
184
+ callback(err);
185
+ }
186
+ } else {
187
+ // Output stream case we want a callback from process
188
+ // and the stream.
189
+ if (outputStream && count === 2) {
190
+ callback();
191
+ } else if (!outputStream && count === 1) {
192
+ callback();
193
+ }
87
194
  }
195
+ }
88
196
 
89
- // wait until the queue size falls to a reasonable level
90
- async.until(
91
- // wait until the queue length drops to twice the paralellism
92
- // or until empty on the last write
93
- function(callback) {
94
- // if we encountered an error, stop this until loop
95
- if (didError) {
96
- return callback(null, true);
97
- }
98
- if (flush) {
99
- callback(null, q.idle() && q.length() === 0);
197
+ // Note use spawn not fork for stdio options not supported with fork in Node 4.x
198
+ backup = spawn('node', args, { stdio: ['ignore', destination, 'pipe'] })
199
+ .on('error', function(err) {
200
+ gatingCallback(err);
201
+ })
202
+ .on('close', function(code, signal) {
203
+ console.log(`Backup process close ${code} ${signal}`);
204
+ try {
205
+ if (params.abort) {
206
+ // The tail should be stopped when we match a line and abort, but if
207
+ // something didn't work we need to make sure the tail is stopped
208
+ tail.unwatch();
209
+ // Assert that the process was aborted as expected
210
+ assert.strictEqual(signal, 'SIGTERM', `The backup should have terminated with SIGTERM, but was ${signal}.`);
211
+ } else if (params.expectedBackupError) {
212
+ assert.strictEqual(code, params.expectedBackupError.code, `The backup exited with unexpected code ${code}.`);
100
213
  } else {
101
- callback(null, q.length() <= parallelism * 2);
102
- }
103
- },
104
- function(cb) {
105
- setTimeout(cb, 20);
106
- },
107
-
108
- function() {
109
- if (flush && !didError) {
110
- writer.emit('finished', { total: written });
214
+ assert.strictEqual(code, 0, `The backup should exit normally, got exit code ${code} and signal ${signal}.`);
111
215
  }
112
- // callback when we're happy with the queue size
113
- callback();
114
- });
216
+ gatingCallback();
217
+ } catch (err) {
218
+ gatingCallback(err);
219
+ }
220
+ });
221
+ // Pipe the stdout to the supplied outputStream
222
+ if (destination === 'pipe') {
223
+ backup.stdout.pipe(backupStream);
224
+ }
225
+
226
+ // Forward the spawned process stderr (we don't use inherit because we want
227
+ // to access this stream directly as well)
228
+ backup.stderr.on('data', function(data) {
229
+ console.error(`${data}`);
230
+ });
231
+
232
+ // Check for errors on the spawned processes
233
+ if (gzip) {
234
+ gzip.on('close', function(code) {
235
+ try {
236
+ assert.strictEqual(code, 0, `The compression should exit normally, got exit code ${code}.`);
237
+ } catch (err) {
238
+ gatingCallback(err);
239
+ }
240
+ });
241
+ }
242
+ if (openssl) {
243
+ openssl.on('close', function(code) {
244
+ try {
245
+ assert.strictEqual(code, 0, `The encryption should exit normally, got exit code ${code}.`);
246
+ } catch (err) {
247
+ gatingCallback(err);
248
+ }
249
+ });
250
+ }
251
+ if (outputStream) {
252
+ // Callback when the destination stream closes.
253
+ outputStream.on('close', gatingCallback);
254
+ } else if (!params.opts.output) {
255
+ gatingCallback(new Error('Unexpected test without outputStream or output option.'));
256
+ }
257
+ }
258
+ return backup;
259
+ }
260
+
261
+ function backupAbort(usingApi, backup) {
262
+ setImmediate(function() {
263
+ if (usingApi) {
264
+ // Currently no way to abort an API backup
265
+ console.error('UNSUPPORTED: cannot abort API backups at this time.');
266
+ } else {
267
+ backup.kill();
268
+ }
269
+ });
270
+ }
271
+
272
+ function testRestore(params, inputStream, databaseName, callback) {
273
+ let restoreStream = inputStream;
274
+
275
+ // Configure API key if needed
276
+ augmentParamsWithApiKey(params);
277
+
278
+ // Pipe via decompression if requested
279
+ if (params.compression) {
280
+ if (params.useApi) {
281
+ // If use API use the Node zlib stream
282
+ restoreStream = zlib.createGunzip();
283
+ inputStream.pipe(restoreStream);
284
+ } else {
285
+ // Spawn process for gunzip
286
+ const gunzip = spawn('gunzip', [], { stdio: ['pipe', 'pipe', 'inherit'] });
287
+ // Pipe the streams as needed
288
+ inputStream.pipe(gunzip.stdin);
289
+ restoreStream = gunzip.stdout;
290
+ }
291
+ }
292
+
293
+ // Pipe via decryption if requested
294
+ if (params.encryption) {
295
+ if (params.useApi) {
296
+ callback(new Error('Not implemented: cannot test encrypted API backups at this time.'));
115
297
  } else {
116
- callback();
298
+ // Spawn process for openssl
299
+ const dopenssl = spawn('openssl', ['aes-128-cbc', '-d', '-pass', 'pass:12345'], { stdio: ['pipe', 'pipe', 'inherit'] });
300
+ // Pipe the streams as needed
301
+ inputStream.pipe(dopenssl.stdin);
302
+ restoreStream = dopenssl.stdout;
117
303
  }
118
304
  }
119
305
 
120
- // take an object
121
- writer._transform = function(obj, encoding, done) {
122
- // each obj that arrives here is a line from the backup file
123
- // it should contain an array of objects. The length of the array
124
- // depends on the bufferSize at backup time.
125
- linenumber++;
126
- if (!didError && obj !== '') {
127
- // see if it parses as JSON
306
+ if (params.useApi) {
307
+ app.restore(restoreStream, dbUrl(process.env.COUCH_URL, databaseName), params.opts, function(err, data) {
308
+ if (err) {
309
+ if (params.expectedRestoreError) {
310
+ try {
311
+ assert.strictEqual(err.name, params.expectedRestoreError.name, 'The restore should receive the expected error.');
312
+ err = null;
313
+ } catch (caught) {
314
+ err = caught;
315
+ }
316
+ }
317
+ } else {
318
+ console.log(data);
319
+ }
320
+ callback(err);
321
+ }).on('error', function(err) {
322
+ console.error(`Caught non-fatal error: ${err}`);
323
+ });
324
+ } else {
325
+ // Set up default args
326
+ const args = ['./bin/couchrestore.bin.js', '--db', databaseName];
327
+ if (params.opts) {
328
+ if (params.opts.bufferSize) {
329
+ args.push('--buffer-size');
330
+ args.push(params.opts.bufferSize);
331
+ }
332
+ if (params.opts.parallelism) {
333
+ args.push('--parallelism');
334
+ args.push(params.opts.parallelism);
335
+ }
336
+ if (params.opts.requestTimeout) {
337
+ args.push('--request-timeout');
338
+ args.push(params.opts.requestTimeout);
339
+ }
340
+ if (params.opts.iamApiKey) {
341
+ args.push('--iam-api-key');
342
+ args.push(params.opts.iamApiKey);
343
+ }
344
+ }
345
+
346
+ // Note use spawn not fork for stdio options not supported with fork in Node 4.x
347
+ const restore = spawn('node', args, { stdio: ['pipe', 'inherit', 'inherit'] });
348
+ // Pipe to write the readable inputStream into stdin
349
+ restoreStream.pipe(restore.stdin);
350
+ restore.stdin.on('error', function(err) {
351
+ // Suppress errors that might arise from piping of input streams
352
+ // from the test process to the child process (this appears to be handled
353
+ // gracefully in the shell)
354
+ console.error(`Test stream error code ${err.code}`);
355
+ });
356
+ restore.on('close', function(code) {
128
357
  try {
129
- const arr = JSON.parse(obj);
130
-
131
- // if it's an array with a length
132
- if (typeof arr === 'object' && arr.length > 0) {
133
- // push each document into a buffer
134
- buffer = buffer.concat(arr);
135
-
136
- // pause the stream
137
- // it's likely that the speed with which data can be read from disk
138
- // may exceed the rate it can be written to CouchDB. To prevent
139
- // the whole file being buffered in memory, we pause the stream here.
140
- // it is resumed, when processBuffer calls back and we call done()
141
- this.pause();
142
-
143
- // break the buffer in to bufferSize chunks to be written to the database
144
- processBuffer(false, done);
358
+ if (params.expectedRestoreError) {
359
+ assert.strictEqual(code, params.expectedRestoreError.code, `The backup exited with unexpected code ${code}.`);
360
+ } else {
361
+ assert.strictEqual(code, 0, `The restore should exit normally, got exit code ${code}`);
362
+ }
363
+ callback();
364
+ } catch (err) {
365
+ callback(err);
366
+ }
367
+ });
368
+ restore.on('error', function(err) {
369
+ callback(err);
370
+ });
371
+ }
372
+ }
373
+
374
+ // Serial backup and restore via a file on disk
375
+ function testBackupAndRestoreViaFile(params, srcDb, backupFile, targetDb, callback) {
376
+ testBackupToFile(params, srcDb, backupFile, function(err) {
377
+ if (err) {
378
+ callback(err);
379
+ } else {
380
+ testRestoreFromFile(params, backupFile, targetDb, function(err) {
381
+ if (!err) {
382
+ dbCompare(srcDb, targetDb, callback);
145
383
  } else {
146
- ee.emit('error', new error.BackupError('BackupFileJsonError', `Error on line ${linenumber} of backup file - not an array`));
147
- done();
384
+ callback(err);
148
385
  }
149
- } catch (e) {
150
- ee.emit('error', new error.BackupError('BackupFileJsonError', `Error on line ${linenumber} of backup file - cannot parse as JSON`));
151
- // Could be an incomplete write that was subsequently resumed
152
- done();
386
+ });
387
+ }
388
+ });
389
+ }
390
+
391
+ function testBackupToFile(params, srcDb, backupFile, callback, processCallback) {
392
+ // Open the file for appending if this is a resume
393
+ const output = fs.createWriteStream(backupFile, { flags: (params.opts && params.opts.resume) ? 'a' : 'w' });
394
+ output.on('open', function() {
395
+ const backupProcess = testBackup(params, srcDb, output, function(err) {
396
+ if (err) {
397
+ callback(err);
398
+ } else {
399
+ callback();
400
+ }
401
+ });
402
+ if (processCallback) {
403
+ processCallback(backupProcess);
404
+ }
405
+ });
406
+ }
407
+
408
+ function testRestoreFromFile(params, backupFile, targetDb, callback) {
409
+ const input = fs.createReadStream(backupFile);
410
+ input.on('open', function() {
411
+ testRestore(params, input, targetDb, function(err) {
412
+ if (err) {
413
+ callback(err);
414
+ } else {
415
+ callback();
153
416
  }
417
+ });
418
+ });
419
+ }
420
+
421
+ function testDirectBackupAndRestore(params, srcDb, targetDb, callback) {
422
+ // Allow a 64 MB highWaterMark for the passthrough during testing
423
+ const passthrough = new stream.PassThrough({ highWaterMark: 67108864 });
424
+ testBackupAndRestore(params, srcDb, passthrough, passthrough, targetDb, callback);
425
+ }
426
+
427
+ function testBackupAndRestore(params, srcDb, backupStream, restoreStream, targetDb, callback) {
428
+ testBackup(params, srcDb, backupStream, function(err) {
429
+ if (err) {
430
+ callback(err);
431
+ }
432
+ });
433
+ testRestore(params, restoreStream, targetDb, function(err) {
434
+ if (err) {
435
+ callback(err);
436
+ } else {
437
+ dbCompare(srcDb, targetDb, callback);
438
+ }
439
+ });
440
+ }
441
+
442
+ function assertResumedBackup(params, resumedBackup, restoreCallback) {
443
+ // Validate that the resume backup didn't need to write all the docs
444
+ if (params.useApi) {
445
+ resumedBackup.once('finished', function(summary) {
446
+ assertWrittenFewerThan(summary.total, params.exclusiveMaxExpected, restoreCallback);
447
+ });
448
+ } else {
449
+ // For the CLI case we need to see the output because we don't have
450
+ // the finished event.
451
+ const listener = function(data) {
452
+ const matches = data.toString().match(/.*Finished - Total document revisions written: (\d+).*/);
453
+ if (matches !== null) {
454
+ assertWrittenFewerThan(matches[1], params.exclusiveMaxExpected, restoreCallback);
455
+ resumedBackup.stderr.removeListener('data', listener);
456
+ }
457
+ };
458
+ resumedBackup.stderr.on('data', listener);
459
+ }
460
+ }
461
+
462
+ function testBackupAbortResumeRestore(params, srcDb, backupFile, targetDb, callback) {
463
+ const restore = function(err) {
464
+ if (err) {
465
+ callback(err);
154
466
  } else {
155
- done();
467
+ testRestoreFromFile(params, backupFile, targetDb, function(err) {
468
+ if (err) {
469
+ callback(err);
470
+ } else {
471
+ dbCompare(srcDb, targetDb, callback);
472
+ }
473
+ });
156
474
  }
157
475
  };
158
476
 
159
- // called when we need to flush everything
160
- writer._flush = function(done) {
161
- processBuffer(true, done);
477
+ const resume = function(err) {
478
+ if (err) {
479
+ callback(err);
480
+ }
481
+ // Remove the abort parameter and add the resume parameter
482
+ delete params.abort;
483
+ params.opts.resume = true;
484
+
485
+ // Resume backup and restore to validate it was successful.
486
+ if (params.opts && params.opts.output) {
487
+ const resumedBackup = testBackup(params, srcDb, null, function(err) {
488
+ if (err) {
489
+ callback(err);
490
+ }
491
+ });
492
+ assertResumedBackup(params, resumedBackup, restore);
493
+ } else {
494
+ testBackupToFile(params, srcDb, backupFile, function(err) {
495
+ if (err) {
496
+ callback(err);
497
+ }
498
+ },
499
+ function(backupProcess) {
500
+ assertResumedBackup(params, backupProcess, restore);
501
+ });
502
+ }
162
503
  };
163
- return writer;
504
+
505
+ if (params.opts && params.opts.output) {
506
+ testBackup(params, srcDb, null, resume);
507
+ } else {
508
+ testBackupToFile(params, srcDb, backupFile, resume);
509
+ }
510
+ }
511
+
512
+ function dbCompare(db1Name, db2Name, callback) {
513
+ const client = request.client(process.env.COUCH_BACKEND_URL, {});
514
+ compare.compare(db1Name, db2Name, client.service)
515
+ .then(result => {
516
+ try {
517
+ assert.strictEqual(result, true, 'The database comparison should succeed, but failed');
518
+ callback();
519
+ } catch (err) {
520
+ callback(err);
521
+ }
522
+ })
523
+ .catch(err => callback(err));
524
+ }
525
+
526
+ function sortByIdThenRev(o1, o2) {
527
+ if (o1._id < o2._id) return -1;
528
+ if (o1._id > o2._id) return 1;
529
+ if (o1._rev < o2._rev) return -1;
530
+ if (o1._rev > o2._rev) return 1;
531
+ return 0;
532
+ }
533
+
534
+ function readSortAndDeepEqual(actualContentPath, expectedContentPath, callback) {
535
+ const backupContent = JSON.parse(fs.readFileSync(actualContentPath, 'utf8'));
536
+ const expectedContent = JSON.parse(fs.readFileSync(expectedContentPath, 'utf8'));
537
+ // Array order of the docs is important for equality, but not for backup
538
+ backupContent.sort(sortByIdThenRev);
539
+ expectedContent.sort(sortByIdThenRev);
540
+ // Assert that the backup matches the expected
541
+ try {
542
+ assert.deepStrictEqual(backupContent, expectedContent);
543
+ callback();
544
+ } catch (err) {
545
+ callback(err);
546
+ }
547
+ }
548
+
549
+ function setTimeout(context, timeout) {
550
+ // Increase timeout using TEST_TIMEOUT_MULTIPLIER
551
+ const multiplier = (typeof process.env.TEST_TIMEOUT_MULTIPLIER !== 'undefined') ? parseInt(process.env.TEST_TIMEOUT_MULTIPLIER) : 1;
552
+ timeout *= multiplier;
553
+ // Set the mocha timeout
554
+ context.timeout(timeout * 1000);
555
+ }
556
+
557
+ function assertGzipFile(path, callback) {
558
+ try {
559
+ // 1f 8b is the gzip magic number
560
+ const expectedBytes = Buffer.from([0x1f, 0x8b]);
561
+ const buffer = Buffer.alloc(2);
562
+ const fd = fs.openSync(path, 'r');
563
+ // Read the first two bytes
564
+ fs.readSync(fd, buffer, 0, 2, 0);
565
+ fs.closeSync(fd);
566
+ // Assert the magic number corresponds to gz extension
567
+ assert.deepStrictEqual(buffer, expectedBytes, 'The backup file should be gz compressed.');
568
+ callback();
569
+ } catch (err) {
570
+ callback(err);
571
+ }
572
+ }
573
+
574
+ function assertEncryptedFile(path, callback) {
575
+ try {
576
+ // Openssl encrypted files start with Salted
577
+ const expectedBytes = Buffer.from('Salted');
578
+ const buffer = Buffer.alloc(6);
579
+ const fd = fs.openSync(path, 'r');
580
+ // Read the first six bytes
581
+ fs.readSync(fd, buffer, 0, 6, 0);
582
+ fs.closeSync(fd);
583
+ // Assert first 6 characters of the file are "Salted"
584
+ assert.deepStrictEqual(buffer, expectedBytes, 'The backup file should be encrypted.');
585
+ callback();
586
+ } catch (err) {
587
+ callback(err);
588
+ }
589
+ }
590
+
591
+ function assertWrittenFewerThan(total, number, callback) {
592
+ try {
593
+ assert(total < number && total > 0, `Saw ${total} but expected between 1 and ${number - 1} documents for the resumed backup.`);
594
+ callback();
595
+ } catch (err) {
596
+ callback(err);
597
+ }
598
+ }
599
+
600
+ function augmentParamsWithApiKey(params) {
601
+ if (process.env.COUCHBACKUP_TEST_IAM_API_KEY) {
602
+ if (!params.opts) {
603
+ params.opts = {};
604
+ }
605
+ params.opts.iamApiKey = process.env.COUCHBACKUP_TEST_IAM_API_KEY;
606
+ params.opts.iamTokenUrl = process.env.CLOUDANT_IAM_TOKEN_URL;
607
+ }
608
+ }
609
+
610
+ module.exports = {
611
+ scenario,
612
+ p: params,
613
+ setTimeout,
614
+ dbCompare,
615
+ readSortAndDeepEqual,
616
+ assertGzipFile,
617
+ assertEncryptedFile,
618
+ testBackup,
619
+ testRestore,
620
+ testDirectBackupAndRestore,
621
+ testBackupToFile,
622
+ testRestoreFromFile,
623
+ testBackupAndRestoreViaFile,
624
+ testBackupAbortResumeRestore
164
625
  };