@cloudant/couchbackup 2.9.12-SNAPSHOT.136 → 2.9.12

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 (414) 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/changesets-10.pb +2 -0
  4. package/.scannerwork/scanner-report/changesets-17.pb +1 -0
  5. package/.scannerwork/scanner-report/changesets-18.pb +1 -0
  6. package/.scannerwork/scanner-report/changesets-19.pb +1 -0
  7. package/.scannerwork/scanner-report/changesets-20.pb +1 -0
  8. package/.scannerwork/scanner-report/changesets-22.pb +1 -0
  9. package/.scannerwork/scanner-report/changesets-23.pb +1 -0
  10. package/.scannerwork/scanner-report/changesets-24.pb +1 -0
  11. package/.scannerwork/scanner-report/changesets-25.pb +1 -0
  12. package/.scannerwork/scanner-report/changesets-26.pb +1 -0
  13. package/.scannerwork/scanner-report/changesets-27.pb +1 -0
  14. package/.scannerwork/scanner-report/changesets-45.pb +1 -0
  15. package/.scannerwork/scanner-report/changesets-46.pb +1 -0
  16. package/.scannerwork/scanner-report/changesets-48.pb +1 -0
  17. package/.scannerwork/scanner-report/changesets-5.pb +1 -0
  18. package/.scannerwork/scanner-report/changesets-64.pb +1 -0
  19. package/.scannerwork/scanner-report/changesets-7.pb +1 -0
  20. package/.scannerwork/scanner-report/changesets-9.pb +1 -0
  21. package/.scannerwork/scanner-report/component-1.pb +2 -1
  22. package/.scannerwork/scanner-report/component-10.pb +2 -0
  23. package/.scannerwork/scanner-report/component-11.pb +1 -1
  24. package/.scannerwork/scanner-report/component-12.pb +1 -1
  25. package/.scannerwork/scanner-report/component-13.pb +1 -1
  26. package/.scannerwork/scanner-report/component-14.pb +1 -1
  27. package/.scannerwork/scanner-report/component-15.pb +1 -1
  28. package/.scannerwork/scanner-report/component-16.pb +1 -1
  29. package/.scannerwork/scanner-report/component-17.pb +1 -0
  30. package/.scannerwork/scanner-report/component-18.pb +1 -0
  31. package/.scannerwork/scanner-report/component-19.pb +1 -0
  32. package/.scannerwork/scanner-report/component-2.pb +1 -0
  33. package/.scannerwork/scanner-report/component-20.pb +1 -0
  34. package/.scannerwork/scanner-report/component-22.pb +1 -0
  35. package/.scannerwork/scanner-report/component-23.pb +1 -0
  36. package/.scannerwork/scanner-report/component-24.pb +1 -0
  37. package/.scannerwork/scanner-report/component-25.pb +1 -0
  38. package/.scannerwork/scanner-report/component-26.pb +1 -0
  39. package/.scannerwork/scanner-report/component-27.pb +1 -0
  40. package/.scannerwork/scanner-report/component-28.pb +1 -1
  41. package/.scannerwork/scanner-report/component-29.pb +1 -1
  42. package/.scannerwork/scanner-report/component-30.pb +1 -1
  43. package/.scannerwork/scanner-report/component-42.pb +1 -1
  44. package/.scannerwork/scanner-report/component-43.pb +1 -1
  45. package/.scannerwork/scanner-report/component-44.pb +1 -1
  46. package/.scannerwork/scanner-report/component-45.pb +1 -0
  47. package/.scannerwork/scanner-report/component-46.pb +1 -0
  48. package/.scannerwork/scanner-report/component-47.pb +1 -1
  49. package/.scannerwork/scanner-report/component-48.pb +1 -0
  50. package/.scannerwork/scanner-report/component-49.pb +1 -1
  51. package/.scannerwork/scanner-report/component-5.pb +1 -0
  52. package/.scannerwork/scanner-report/component-50.pb +1 -1
  53. package/.scannerwork/scanner-report/component-51.pb +1 -1
  54. package/.scannerwork/scanner-report/component-52.pb +1 -1
  55. package/.scannerwork/scanner-report/component-53.pb +1 -1
  56. package/.scannerwork/scanner-report/component-54.pb +1 -1
  57. package/.scannerwork/scanner-report/component-55.pb +1 -1
  58. package/.scannerwork/scanner-report/component-56.pb +1 -1
  59. package/.scannerwork/scanner-report/component-58.pb +1 -1
  60. package/.scannerwork/scanner-report/component-6.pb +1 -1
  61. package/.scannerwork/scanner-report/component-61.pb +1 -1
  62. package/.scannerwork/scanner-report/component-64.pb +1 -1
  63. package/.scannerwork/scanner-report/component-68.pb +1 -1
  64. package/.scannerwork/scanner-report/component-7.pb +1 -0
  65. package/.scannerwork/scanner-report/component-8.pb +1 -1
  66. package/.scannerwork/scanner-report/component-9.pb +1 -0
  67. package/.scannerwork/scanner-report/coverages-10.pb +0 -0
  68. package/.scannerwork/scanner-report/coverages-11.pb +0 -0
  69. package/.scannerwork/scanner-report/coverages-12.pb +0 -0
  70. package/.scannerwork/scanner-report/coverages-13.pb +0 -0
  71. package/.scannerwork/scanner-report/coverages-14.pb +0 -0
  72. package/.scannerwork/scanner-report/coverages-15.pb +0 -0
  73. package/.scannerwork/scanner-report/coverages-16.pb +0 -0
  74. package/.scannerwork/scanner-report/coverages-18.pb +0 -0
  75. package/.scannerwork/scanner-report/coverages-19.pb +0 -0
  76. package/.scannerwork/scanner-report/coverages-24.pb +0 -0
  77. package/.scannerwork/scanner-report/{coverages-34.pb → coverages-25.pb} +0 -0
  78. package/.scannerwork/scanner-report/coverages-28.pb +0 -0
  79. package/.scannerwork/scanner-report/coverages-29.pb +0 -0
  80. package/.scannerwork/scanner-report/coverages-30.pb +0 -0
  81. package/.scannerwork/scanner-report/coverages-42.pb +0 -0
  82. package/.scannerwork/scanner-report/coverages-43.pb +0 -0
  83. package/.scannerwork/scanner-report/coverages-44.pb +0 -0
  84. package/.scannerwork/scanner-report/coverages-45.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-53.pb +0 -0
  93. package/.scannerwork/scanner-report/coverages-54.pb +0 -0
  94. package/.scannerwork/scanner-report/coverages-55.pb +0 -0
  95. package/.scannerwork/scanner-report/coverages-56.pb +0 -0
  96. package/.scannerwork/scanner-report/coverages-58.pb +0 -0
  97. package/.scannerwork/scanner-report/coverages-6.pb +0 -0
  98. package/.scannerwork/scanner-report/coverages-64.pb +0 -0
  99. package/.scannerwork/scanner-report/coverages-7.pb +0 -0
  100. package/.scannerwork/scanner-report/coverages-8.pb +0 -0
  101. package/.scannerwork/scanner-report/duplications-12.pb +0 -2
  102. package/.scannerwork/scanner-report/duplications-15.pb +2 -0
  103. package/.scannerwork/scanner-report/duplications-24.pb +2 -0
  104. package/.scannerwork/scanner-report/duplications-26.pb +3 -0
  105. package/.scannerwork/scanner-report/duplications-28.pb +0 -2
  106. package/.scannerwork/scanner-report/duplications-29.pb +2 -0
  107. package/.scannerwork/scanner-report/duplications-30.pb +3 -0
  108. package/.scannerwork/scanner-report/duplications-5.pb +2 -0
  109. package/.scannerwork/scanner-report/duplications-52.pb +3 -0
  110. package/.scannerwork/scanner-report/duplications-55.pb +2 -0
  111. package/.scannerwork/scanner-report/duplications-6.pb +1 -1
  112. package/.scannerwork/scanner-report/duplications-64.pb +0 -0
  113. package/.scannerwork/scanner-report/duplications-7.pb +0 -0
  114. package/.scannerwork/scanner-report/duplications-8.pb +0 -2
  115. package/.scannerwork/scanner-report/duplications-9.pb +2 -0
  116. package/.scannerwork/scanner-report/measures-10.pb +0 -0
  117. package/.scannerwork/scanner-report/measures-11.pb +0 -0
  118. package/.scannerwork/scanner-report/measures-12.pb +0 -0
  119. package/.scannerwork/scanner-report/measures-13.pb +0 -0
  120. package/.scannerwork/scanner-report/measures-14.pb +0 -0
  121. package/.scannerwork/scanner-report/measures-15.pb +0 -0
  122. package/.scannerwork/scanner-report/measures-16.pb +0 -0
  123. package/.scannerwork/scanner-report/measures-18.pb +16 -0
  124. package/.scannerwork/scanner-report/measures-19.pb +0 -0
  125. package/.scannerwork/scanner-report/measures-20.pb +0 -0
  126. package/.scannerwork/scanner-report/measures-23.pb +0 -0
  127. package/.scannerwork/scanner-report/measures-24.pb +0 -0
  128. package/.scannerwork/scanner-report/measures-25.pb +0 -0
  129. package/.scannerwork/scanner-report/measures-28.pb +0 -0
  130. package/.scannerwork/scanner-report/measures-29.pb +0 -0
  131. package/.scannerwork/scanner-report/measures-30.pb +0 -0
  132. package/.scannerwork/scanner-report/measures-42.pb +0 -0
  133. package/.scannerwork/scanner-report/measures-43.pb +0 -0
  134. package/.scannerwork/scanner-report/measures-44.pb +0 -0
  135. package/.scannerwork/scanner-report/measures-45.pb +0 -0
  136. package/.scannerwork/scanner-report/measures-47.pb +0 -0
  137. package/.scannerwork/scanner-report/measures-48.pb +0 -0
  138. package/.scannerwork/scanner-report/measures-49.pb +0 -0
  139. package/.scannerwork/scanner-report/measures-5.pb +0 -0
  140. package/.scannerwork/scanner-report/measures-50.pb +0 -0
  141. package/.scannerwork/scanner-report/measures-51.pb +0 -0
  142. package/.scannerwork/scanner-report/measures-52.pb +0 -0
  143. package/.scannerwork/scanner-report/measures-53.pb +0 -0
  144. package/.scannerwork/scanner-report/measures-54.pb +0 -0
  145. package/.scannerwork/scanner-report/measures-55.pb +0 -0
  146. package/.scannerwork/scanner-report/measures-56.pb +0 -0
  147. package/.scannerwork/scanner-report/measures-58.pb +0 -0
  148. package/.scannerwork/scanner-report/measures-6.pb +0 -0
  149. package/.scannerwork/scanner-report/measures-61.pb +0 -0
  150. package/.scannerwork/scanner-report/measures-64.pb +0 -0
  151. package/.scannerwork/scanner-report/measures-68.pb +0 -0
  152. package/.scannerwork/scanner-report/measures-7.pb +0 -0
  153. package/.scannerwork/scanner-report/measures-8.pb +0 -0
  154. package/.scannerwork/scanner-report/metadata.pb +0 -0
  155. package/.scannerwork/scanner-report/source-10.txt +111 -0
  156. package/.scannerwork/scanner-report/source-11.txt +35 -81
  157. package/.scannerwork/scanner-report/source-12.txt +22 -126
  158. package/.scannerwork/scanner-report/source-13.txt +115 -54
  159. package/.scannerwork/scanner-report/source-14.txt +149 -21
  160. package/.scannerwork/scanner-report/source-15.txt +61 -352
  161. package/.scannerwork/scanner-report/source-16.txt +104 -22
  162. package/.scannerwork/scanner-report/source-18.txt +178 -0
  163. package/.scannerwork/scanner-report/source-19.txt +31 -0
  164. package/.scannerwork/scanner-report/{source-4.txt → source-2.txt} +141 -141
  165. package/.scannerwork/scanner-report/source-20.txt +281 -0
  166. package/.scannerwork/scanner-report/source-23.txt +77 -0
  167. package/.scannerwork/scanner-report/source-24.txt +145 -0
  168. package/.scannerwork/scanner-report/{source-34.txt → source-25.txt} +9 -10
  169. package/.scannerwork/scanner-report/source-28.txt +25 -203
  170. package/.scannerwork/scanner-report/source-29.txt +265 -64
  171. package/.scannerwork/scanner-report/source-30.txt +104 -36
  172. package/.scannerwork/scanner-report/source-42.txt +346 -14
  173. package/.scannerwork/scanner-report/source-43.txt +120 -115
  174. package/.scannerwork/scanner-report/source-44.txt +20 -403
  175. package/.scannerwork/scanner-report/source-45.txt +142 -0
  176. package/.scannerwork/scanner-report/source-47.txt +17 -95
  177. package/.scannerwork/scanner-report/{source-41.txt → source-48.txt} +13 -6
  178. package/.scannerwork/scanner-report/source-49.txt +85 -65
  179. package/.scannerwork/scanner-report/source-5.txt +175 -0
  180. package/.scannerwork/scanner-report/source-50.txt +602 -106
  181. package/.scannerwork/scanner-report/source-51.txt +51 -14
  182. package/.scannerwork/scanner-report/source-52.txt +225 -22
  183. package/.scannerwork/scanner-report/source-53.txt +45 -160
  184. package/.scannerwork/scanner-report/source-54.txt +12 -58
  185. package/.scannerwork/scanner-report/source-55.txt +208 -37
  186. package/.scannerwork/scanner-report/source-56.txt +78 -32
  187. package/.scannerwork/scanner-report/source-58.txt +94 -91
  188. package/.scannerwork/scanner-report/source-6.txt +64 -49
  189. package/.scannerwork/scanner-report/source-61.txt +509 -281
  190. package/.scannerwork/scanner-report/source-64.txt +418 -167
  191. package/.scannerwork/scanner-report/source-68.txt +167 -77
  192. package/.scannerwork/scanner-report/source-7.txt +46 -0
  193. package/.scannerwork/scanner-report/source-8.txt +65 -175
  194. package/.scannerwork/scanner-report/symbols-10.pb +59 -0
  195. package/.scannerwork/scanner-report/symbols-11.pb +14 -68
  196. package/.scannerwork/scanner-report/symbols-12.pb +18 -102
  197. package/.scannerwork/scanner-report/symbols-13.pb +33 -36
  198. package/.scannerwork/scanner-report/symbols-14.pb +126 -14
  199. package/.scannerwork/scanner-report/symbols-15.pb +42 -494
  200. package/.scannerwork/scanner-report/symbols-16.pb +30 -11
  201. package/.scannerwork/scanner-report/symbols-18.pb +144 -0
  202. package/.scannerwork/scanner-report/symbols-19.pb +14 -0
  203. package/.scannerwork/scanner-report/symbols-24.pb +102 -0
  204. package/.scannerwork/scanner-report/symbols-25.pb +10 -0
  205. package/.scannerwork/scanner-report/symbols-28.pb +26 -417
  206. package/.scannerwork/scanner-report/symbols-29.pb +486 -60
  207. package/.scannerwork/scanner-report/symbols-30.pb +97 -43
  208. package/.scannerwork/scanner-report/symbols-42.pb +494 -10
  209. package/.scannerwork/scanner-report/symbols-43.pb +80 -111
  210. package/.scannerwork/scanner-report/symbols-44.pb +17 -610
  211. package/.scannerwork/scanner-report/symbols-45.pb +116 -0
  212. package/.scannerwork/scanner-report/symbols-47.pb +14 -28
  213. package/.scannerwork/scanner-report/symbols-48.pb +11 -0
  214. package/.scannerwork/scanner-report/symbols-49.pb +68 -45
  215. package/.scannerwork/scanner-report/symbols-5.pb +153 -0
  216. package/.scannerwork/scanner-report/symbols-50.pb +1254 -33
  217. package/.scannerwork/scanner-report/symbols-51.pb +35 -13
  218. package/.scannerwork/scanner-report/symbols-52.pb +393 -17
  219. package/.scannerwork/scanner-report/symbols-53.pb +42 -143
  220. package/.scannerwork/scanner-report/symbols-54.pb +9 -42
  221. package/.scannerwork/scanner-report/symbols-55.pb +417 -14
  222. package/.scannerwork/scanner-report/symbols-56.pb +60 -20
  223. package/.scannerwork/scanner-report/symbols-58.pb +28 -59
  224. package/.scannerwork/scanner-report/symbols-6.pb +176 -149
  225. package/.scannerwork/scanner-report/symbols-64.pb +610 -0
  226. package/.scannerwork/scanner-report/symbols-7.pb +20 -0
  227. package/.scannerwork/scanner-report/symbols-8.pb +46 -180
  228. package/.scannerwork/scanner-report/syntax-highlightings-10.pb +133 -0
  229. package/.scannerwork/scanner-report/syntax-highlightings-11.pb +24 -75
  230. package/.scannerwork/scanner-report/syntax-highlightings-12.pb +21 -198
  231. package/.scannerwork/scanner-report/syntax-highlightings-13.pb +93 -59
  232. package/.scannerwork/scanner-report/syntax-highlightings-14.pb +180 -21
  233. package/.scannerwork/scanner-report/syntax-highlightings-15.pb +46 -598
  234. package/.scannerwork/scanner-report/syntax-highlightings-16.pb +84 -30
  235. package/.scannerwork/scanner-report/syntax-highlightings-18.pb +243 -0
  236. package/.scannerwork/scanner-report/syntax-highlightings-19.pb +45 -0
  237. package/.scannerwork/scanner-report/{syntax-highlightings-4.pb → syntax-highlightings-2.pb} +88 -86
  238. package/.scannerwork/scanner-report/syntax-highlightings-20.pb +285 -0
  239. package/.scannerwork/scanner-report/syntax-highlightings-23.pb +103 -0
  240. package/.scannerwork/scanner-report/syntax-highlightings-24.pb +227 -0
  241. package/.scannerwork/scanner-report/{syntax-highlightings-41.pb → syntax-highlightings-25.pb} +15 -13
  242. package/.scannerwork/scanner-report/syntax-highlightings-28.pb +28 -291
  243. package/.scannerwork/scanner-report/syntax-highlightings-29.pb +615 -66
  244. package/.scannerwork/scanner-report/syntax-highlightings-30.pb +143 -43
  245. package/.scannerwork/scanner-report/syntax-highlightings-42.pb +594 -14
  246. package/.scannerwork/scanner-report/syntax-highlightings-43.pb +154 -97
  247. package/.scannerwork/scanner-report/syntax-highlightings-44.pb +34 -619
  248. package/.scannerwork/scanner-report/syntax-highlightings-45.pb +134 -0
  249. package/.scannerwork/scanner-report/syntax-highlightings-47.pb +20 -104
  250. package/.scannerwork/scanner-report/syntax-highlightings-48.pb +59 -0
  251. package/.scannerwork/scanner-report/syntax-highlightings-49.pb +71 -61
  252. package/.scannerwork/scanner-report/syntax-highlightings-5.pb +214 -0
  253. package/.scannerwork/scanner-report/syntax-highlightings-50.pb +847 -89
  254. package/.scannerwork/scanner-report/syntax-highlightings-51.pb +65 -15
  255. package/.scannerwork/scanner-report/syntax-highlightings-52.pb +300 -21
  256. package/.scannerwork/scanner-report/syntax-highlightings-53.pb +46 -207
  257. package/.scannerwork/scanner-report/syntax-highlightings-54.pb +15 -45
  258. package/.scannerwork/scanner-report/syntax-highlightings-55.pb +301 -24
  259. package/.scannerwork/scanner-report/syntax-highlightings-56.pb +78 -32
  260. package/.scannerwork/scanner-report/syntax-highlightings-58.pb +103 -98
  261. package/.scannerwork/scanner-report/syntax-highlightings-6.pb +138 -127
  262. package/.scannerwork/scanner-report/syntax-highlightings-61.pb +3662 -199
  263. package/.scannerwork/scanner-report/syntax-highlightings-64.pb +509 -809
  264. package/.scannerwork/scanner-report/syntax-highlightings-68.pb +917 -71
  265. package/.scannerwork/scanner-report/syntax-highlightings-7.pb +64 -0
  266. package/.scannerwork/scanner-report/syntax-highlightings-8.pb +64 -194
  267. package/.secrets.baseline +2 -2
  268. package/CHANGES.md +3 -0
  269. package/package.json +1 -1
  270. package/test-16-results.xml +152 -152
  271. package/test-18-results.xml +150 -150
  272. package/test-iam-18-results.xml +50 -50
  273. package/.scannerwork/scanner-report/changesets-31.pb +0 -1
  274. package/.scannerwork/scanner-report/changesets-32.pb +0 -1
  275. package/.scannerwork/scanner-report/changesets-33.pb +0 -1
  276. package/.scannerwork/scanner-report/changesets-34.pb +0 -1
  277. package/.scannerwork/scanner-report/changesets-35.pb +0 -1
  278. package/.scannerwork/scanner-report/changesets-36.pb +0 -1
  279. package/.scannerwork/scanner-report/changesets-37.pb +0 -1
  280. package/.scannerwork/scanner-report/changesets-38.pb +0 -1
  281. package/.scannerwork/scanner-report/changesets-39.pb +0 -1
  282. package/.scannerwork/scanner-report/changesets-40.pb +0 -1
  283. package/.scannerwork/scanner-report/changesets-41.pb +0 -1
  284. package/.scannerwork/scanner-report/changesets-57.pb +0 -1
  285. package/.scannerwork/scanner-report/changesets-59.pb +0 -1
  286. package/.scannerwork/scanner-report/changesets-60.pb +0 -1
  287. package/.scannerwork/scanner-report/changesets-61.pb +0 -1
  288. package/.scannerwork/scanner-report/changesets-62.pb +0 -1
  289. package/.scannerwork/scanner-report/changesets-67.pb +0 -1
  290. package/.scannerwork/scanner-report/changesets-68.pb +0 -1
  291. package/.scannerwork/scanner-report/component-31.pb +0 -1
  292. package/.scannerwork/scanner-report/component-32.pb +0 -1
  293. package/.scannerwork/scanner-report/component-33.pb +0 -1
  294. package/.scannerwork/scanner-report/component-34.pb +0 -1
  295. package/.scannerwork/scanner-report/component-35.pb +0 -1
  296. package/.scannerwork/scanner-report/component-36.pb +0 -1
  297. package/.scannerwork/scanner-report/component-37.pb +0 -1
  298. package/.scannerwork/scanner-report/component-38.pb +0 -1
  299. package/.scannerwork/scanner-report/component-39.pb +0 -1
  300. package/.scannerwork/scanner-report/component-4.pb +0 -1
  301. package/.scannerwork/scanner-report/component-40.pb +0 -1
  302. package/.scannerwork/scanner-report/component-41.pb +0 -1
  303. package/.scannerwork/scanner-report/component-57.pb +0 -1
  304. package/.scannerwork/scanner-report/component-59.pb +0 -1
  305. package/.scannerwork/scanner-report/component-60.pb +0 -1
  306. package/.scannerwork/scanner-report/component-62.pb +0 -1
  307. package/.scannerwork/scanner-report/component-65.pb +0 -1
  308. package/.scannerwork/scanner-report/component-67.pb +0 -1
  309. package/.scannerwork/scanner-report/coverages-31.pb +0 -0
  310. package/.scannerwork/scanner-report/coverages-32.pb +0 -0
  311. package/.scannerwork/scanner-report/coverages-33.pb +0 -0
  312. package/.scannerwork/scanner-report/coverages-37.pb +0 -0
  313. package/.scannerwork/scanner-report/coverages-38.pb +0 -0
  314. package/.scannerwork/scanner-report/coverages-40.pb +0 -0
  315. package/.scannerwork/scanner-report/coverages-41.pb +0 -0
  316. package/.scannerwork/scanner-report/coverages-57.pb +0 -0
  317. package/.scannerwork/scanner-report/coverages-60.pb +0 -0
  318. package/.scannerwork/scanner-report/duplications-31.pb +0 -2
  319. package/.scannerwork/scanner-report/duplications-33.pb +0 -3
  320. package/.scannerwork/scanner-report/duplications-36.pb +0 -3
  321. package/.scannerwork/scanner-report/duplications-38.pb +0 -3
  322. package/.scannerwork/scanner-report/duplications-54.pb +0 -2
  323. package/.scannerwork/scanner-report/duplications-59.pb +0 -2
  324. package/.scannerwork/scanner-report/measures-31.pb +0 -15
  325. package/.scannerwork/scanner-report/measures-32.pb +0 -0
  326. package/.scannerwork/scanner-report/measures-33.pb +0 -0
  327. package/.scannerwork/scanner-report/measures-34.pb +0 -0
  328. package/.scannerwork/scanner-report/measures-37.pb +0 -0
  329. package/.scannerwork/scanner-report/measures-38.pb +0 -0
  330. package/.scannerwork/scanner-report/measures-40.pb +0 -0
  331. package/.scannerwork/scanner-report/measures-41.pb +0 -0
  332. package/.scannerwork/scanner-report/measures-57.pb +0 -0
  333. package/.scannerwork/scanner-report/measures-60.pb +0 -0
  334. package/.scannerwork/scanner-report/measures-65.pb +0 -0
  335. package/.scannerwork/scanner-report/source-31.txt +0 -293
  336. package/.scannerwork/scanner-report/source-32.txt +0 -625
  337. package/.scannerwork/scanner-report/source-33.txt +0 -244
  338. package/.scannerwork/scanner-report/source-37.txt +0 -147
  339. package/.scannerwork/scanner-report/source-38.txt +0 -131
  340. package/.scannerwork/scanner-report/source-40.txt +0 -47
  341. package/.scannerwork/scanner-report/source-57.txt +0 -164
  342. package/.scannerwork/scanner-report/source-60.txt +0 -118
  343. package/.scannerwork/scanner-report/source-65.txt +0 -509
  344. package/.scannerwork/scanner-report/symbols-31.pb +0 -486
  345. package/.scannerwork/scanner-report/symbols-32.pb +0 -1254
  346. package/.scannerwork/scanner-report/symbols-33.pb +0 -394
  347. package/.scannerwork/scanner-report/symbols-34.pb +0 -17
  348. package/.scannerwork/scanner-report/symbols-37.pb +0 -85
  349. package/.scannerwork/scanner-report/symbols-38.pb +0 -97
  350. package/.scannerwork/scanner-report/symbols-40.pb +0 -27
  351. package/.scannerwork/scanner-report/symbols-41.pb +0 -9
  352. package/.scannerwork/scanner-report/symbols-57.pb +0 -126
  353. package/.scannerwork/scanner-report/symbols-60.pb +0 -30
  354. package/.scannerwork/scanner-report/syntax-highlightings-31.pb +0 -659
  355. package/.scannerwork/scanner-report/syntax-highlightings-32.pb +0 -887
  356. package/.scannerwork/scanner-report/syntax-highlightings-33.pb +0 -329
  357. package/.scannerwork/scanner-report/syntax-highlightings-34.pb +0 -63
  358. package/.scannerwork/scanner-report/syntax-highlightings-37.pb +0 -191
  359. package/.scannerwork/scanner-report/syntax-highlightings-38.pb +0 -182
  360. package/.scannerwork/scanner-report/syntax-highlightings-40.pb +0 -68
  361. package/.scannerwork/scanner-report/syntax-highlightings-57.pb +0 -213
  362. package/.scannerwork/scanner-report/syntax-highlightings-60.pb +0 -113
  363. package/.scannerwork/scanner-report/syntax-highlightings-65.pb +0 -3746
  364. /package/.scannerwork/scanner-report/{coverages-62.pb → coverages-17.pb} +0 -0
  365. /package/.scannerwork/scanner-report/{coverages-61.pb → coverages-20.pb} +0 -0
  366. /package/.scannerwork/scanner-report/{coverages-67.pb → coverages-22.pb} +0 -0
  367. /package/.scannerwork/scanner-report/{coverages-68.pb → coverages-23.pb} +0 -0
  368. /package/.scannerwork/scanner-report/{coverages-36.pb → coverages-26.pb} +0 -0
  369. /package/.scannerwork/scanner-report/{coverages-39.pb → coverages-27.pb} +0 -0
  370. /package/.scannerwork/scanner-report/{coverages-35.pb → coverages-46.pb} +0 -0
  371. /package/.scannerwork/scanner-report/{coverages-59.pb → coverages-9.pb} +0 -0
  372. /package/.scannerwork/scanner-report/{duplications-32.pb → duplications-10.pb} +0 -0
  373. /package/.scannerwork/scanner-report/{duplications-34.pb → duplications-17.pb} +0 -0
  374. /package/.scannerwork/scanner-report/{duplications-35.pb → duplications-18.pb} +0 -0
  375. /package/.scannerwork/scanner-report/{duplications-37.pb → duplications-19.pb} +0 -0
  376. /package/.scannerwork/scanner-report/{duplications-39.pb → duplications-20.pb} +0 -0
  377. /package/.scannerwork/scanner-report/{duplications-40.pb → duplications-22.pb} +0 -0
  378. /package/.scannerwork/scanner-report/{duplications-57.pb → duplications-23.pb} +0 -0
  379. /package/.scannerwork/scanner-report/{duplications-60.pb → duplications-25.pb} +0 -0
  380. /package/.scannerwork/scanner-report/{duplications-61.pb → duplications-27.pb} +0 -0
  381. /package/.scannerwork/scanner-report/{duplications-62.pb → duplications-45.pb} +0 -0
  382. /package/.scannerwork/scanner-report/{duplications-67.pb → duplications-46.pb} +0 -0
  383. /package/.scannerwork/scanner-report/{duplications-68.pb → duplications-48.pb} +0 -0
  384. /package/.scannerwork/scanner-report/{issues-40.pb → issues-28.pb} +0 -0
  385. /package/.scannerwork/scanner-report/{issues-15.pb → issues-42.pb} +0 -0
  386. /package/.scannerwork/scanner-report/{issues-43.pb → issues-45.pb} +0 -0
  387. /package/.scannerwork/scanner-report/{issues-33.pb → issues-52.pb} +0 -0
  388. /package/.scannerwork/scanner-report/{measures-62.pb → measures-17.pb} +0 -0
  389. /package/.scannerwork/scanner-report/{measures-4.pb → measures-2.pb} +0 -0
  390. /package/.scannerwork/scanner-report/{measures-67.pb → measures-22.pb} +0 -0
  391. /package/.scannerwork/scanner-report/{measures-36.pb → measures-26.pb} +0 -0
  392. /package/.scannerwork/scanner-report/{measures-39.pb → measures-27.pb} +0 -0
  393. /package/.scannerwork/scanner-report/{measures-35.pb → measures-46.pb} +0 -0
  394. /package/.scannerwork/scanner-report/{measures-59.pb → measures-9.pb} +0 -0
  395. /package/.scannerwork/scanner-report/{source-62.txt → source-17.txt} +0 -0
  396. /package/.scannerwork/scanner-report/{source-67.txt → source-22.txt} +0 -0
  397. /package/.scannerwork/scanner-report/{source-36.txt → source-26.txt} +0 -0
  398. /package/.scannerwork/scanner-report/{source-39.txt → source-27.txt} +0 -0
  399. /package/.scannerwork/scanner-report/{source-35.txt → source-46.txt} +0 -0
  400. /package/.scannerwork/scanner-report/{source-59.txt → source-9.txt} +0 -0
  401. /package/.scannerwork/scanner-report/{symbols-62.pb → symbols-17.pb} +0 -0
  402. /package/.scannerwork/scanner-report/{symbols-61.pb → symbols-20.pb} +0 -0
  403. /package/.scannerwork/scanner-report/{symbols-67.pb → symbols-22.pb} +0 -0
  404. /package/.scannerwork/scanner-report/{symbols-68.pb → symbols-23.pb} +0 -0
  405. /package/.scannerwork/scanner-report/{symbols-36.pb → symbols-26.pb} +0 -0
  406. /package/.scannerwork/scanner-report/{symbols-39.pb → symbols-27.pb} +0 -0
  407. /package/.scannerwork/scanner-report/{symbols-35.pb → symbols-46.pb} +0 -0
  408. /package/.scannerwork/scanner-report/{symbols-59.pb → symbols-9.pb} +0 -0
  409. /package/.scannerwork/scanner-report/{syntax-highlightings-62.pb → syntax-highlightings-17.pb} +0 -0
  410. /package/.scannerwork/scanner-report/{syntax-highlightings-67.pb → syntax-highlightings-22.pb} +0 -0
  411. /package/.scannerwork/scanner-report/{syntax-highlightings-36.pb → syntax-highlightings-26.pb} +0 -0
  412. /package/.scannerwork/scanner-report/{syntax-highlightings-39.pb → syntax-highlightings-27.pb} +0 -0
  413. /package/.scannerwork/scanner-report/{syntax-highlightings-35.pb → syntax-highlightings-46.pb} +0 -0
  414. /package/.scannerwork/scanner-report/{syntax-highlightings-59.pb → syntax-highlightings-9.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,119 +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 cliutils = require('./cliutils.js');
17
- const config = require('./config.js');
18
- const error = require('./error.js');
19
- const path = require('path');
20
- const pkg = require('../package.json');
21
-
22
- function parseBackupArgs() {
23
- const program = require('commander');
24
-
25
- // Option CLI defaults
26
- const defaults = config.cliDefaults();
27
-
28
- // Options set by environment variables
29
- const envVarOptions = {};
30
- config.applyEnvironmentVariables(envVarOptions);
31
-
32
- program
33
- .version(pkg.version)
34
- .description('Backup a CouchDB/Cloudant database to a backup text file.')
35
- .usage('[options...]')
36
- .option('-b, --buffer-size <n>',
37
- cliutils.getUsage('number of documents fetched at once', defaults.bufferSize),
38
- Number)
39
- .option('-d, --db <db>',
40
- cliutils.getUsage('name of the database to backup', defaults.db))
41
- .option('-k, --iam-api-key <API key>',
42
- cliutils.getUsage('IAM API key to access the Cloudant server'))
43
- .option('-l, --log <file>',
44
- cliutils.getUsage('file to store logging information during backup; invalid in "shallow" mode', 'a temporary file'),
45
- path.normalize)
46
- .option('-m, --mode <mode>',
47
- cliutils.getUsage('"shallow" if only a superficial backup is done (ignoring conflicts and revision tokens), else "full" for complete backup', defaults.mode),
48
- (mode) => { return mode.toLowerCase(); })
49
- .option('-o, --output <file>',
50
- cliutils.getUsage('file name to store the backup data', 'stdout'),
51
- path.normalize)
52
- .option('-p, --parallelism <n>',
53
- cliutils.getUsage('number of HTTP requests to perform in parallel when performing a backup; ignored in "shallow" mode', defaults.parallelism),
54
- Number)
55
- .option('-q, --quiet',
56
- cliutils.getUsage('suppress batch messages', defaults.quiet))
57
- .option('-r, --resume',
58
- cliutils.getUsage('continue a previous backup from its last known position; invalid in "shallow" mode', defaults.resume))
59
- .option('-t, --request-timeout <n>',
60
- cliutils.getUsage('milliseconds to wait for a response to a HTTP request before retrying the request', defaults.requestTimeout),
61
- Number)
62
- .option('-u, --url <url>',
63
- cliutils.getUsage('URL of the CouchDB/Cloudant server', defaults.url))
64
- .parse(process.argv);
65
-
66
- // Remove defaults that don't apply when using shallow mode
67
- if (program.opts().mode === 'shallow' || envVarOptions.mode === 'shallow') {
68
- delete defaults.parallelism;
69
- delete defaults.log;
70
- delete defaults.resume;
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;
22
+ const stream = require('stream');
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);
67
+ });
68
+ }
71
69
  }
72
70
 
73
- // Apply the options in order so that the CLI overrides env vars and env variables
74
- // override defaults.
75
- const opts = Object.assign({}, defaults, envVarOptions, program.opts());
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
+ }
88
+
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);
95
+
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);
105
+ }
106
+ });
107
+ tail.on('error', function(err) {
108
+ callback(err);
109
+ });
110
+ }
111
+
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';
138
+
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
+ }
168
+
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
+ }
194
+ }
195
+ }
196
+
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}.`);
213
+ } else {
214
+ assert.strictEqual(code, 0, `The backup should exit normally, got exit code ${code} and signal ${signal}.`);
215
+ }
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
+ });
76
231
 
77
- if (opts.resume && (opts.log === defaults.log)) {
78
- // If resuming and the log file arg is the newly generated tmp name from defaults then we know that --log wasn't specified.
79
- // We have to do this check here for the CLI case because of the default.
80
- error.terminationCallback(new error.BackupError('NoLogFileName', 'To resume a backup, a log file must be specified'));
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
+ }
81
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.'));
297
+ } else {
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;
303
+ }
304
+ }
305
+
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) {
357
+ try {
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);
383
+ } else {
384
+ callback(err);
385
+ }
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
+ }
82
407
 
83
- return opts;
84
- }
85
-
86
- function parseRestoreArgs() {
87
- const program = require('commander');
88
-
89
- // Option CLI defaults
90
- const defaults = config.cliDefaults();
91
-
92
- // Options set by environment variables
93
- const envVarOptions = {};
94
- config.applyEnvironmentVariables(envVarOptions);
95
-
96
- program
97
- .version(pkg.version)
98
- .description('Restore a CouchDB/Cloudant database from a backup text file.')
99
- .usage('[options...]')
100
- .option('-b, --buffer-size <n>',
101
- cliutils.getUsage('number of documents restored at once', defaults.bufferSize),
102
- Number)
103
- .option('-d, --db <db>',
104
- cliutils.getUsage('name of the new, existing database to restore to', defaults.db))
105
- .option('-k, --iam-api-key <API key>',
106
- cliutils.getUsage('IAM API key to access the Cloudant server'))
107
- .option('-p, --parallelism <n>',
108
- cliutils.getUsage('number of HTTP requests to perform in parallel when restoring a backup', defaults.parallelism),
109
- Number)
110
- .option('-q, --quiet',
111
- cliutils.getUsage('suppress batch messages', defaults.quiet))
112
- .option('-t, --request-timeout <n>',
113
- cliutils.getUsage('milliseconds to wait for a response to a HTTP request before retrying the request', defaults.requestTimeout),
114
- Number)
115
- .option('-u, --url <url>',
116
- cliutils.getUsage('URL of the CouchDB/Cloudant server', defaults.url))
117
- .parse(process.argv);
118
-
119
- // Apply the options in order so that the CLI overrides env vars and env variables
120
- // override defaults.
121
- const opts = Object.assign({}, defaults, envVarOptions, program.opts());
122
-
123
- return opts;
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();
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);
466
+ } else {
467
+ testRestoreFromFile(params, backupFile, targetDb, function(err) {
468
+ if (err) {
469
+ callback(err);
470
+ } else {
471
+ dbCompare(srcDb, targetDb, callback);
472
+ }
473
+ });
474
+ }
475
+ };
476
+
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
+ }
503
+ };
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
+ }
124
608
  }
125
609
 
126
610
  module.exports = {
127
- parseBackupArgs: parseBackupArgs,
128
- parseRestoreArgs: parseRestoreArgs
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
129
625
  };