@cloudant/couchbackup 2.9.14-SNAPSHOT.159 → 2.9.14

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 (395) 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-18.pb +1 -0
  4. package/.scannerwork/scanner-report/changesets-21.pb +1 -0
  5. package/.scannerwork/scanner-report/changesets-23.pb +1 -0
  6. package/.scannerwork/scanner-report/changesets-24.pb +1 -0
  7. package/.scannerwork/scanner-report/changesets-25.pb +1 -0
  8. package/.scannerwork/scanner-report/changesets-49.pb +1 -0
  9. package/.scannerwork/scanner-report/changesets-50.pb +1 -0
  10. package/.scannerwork/scanner-report/changesets-51.pb +1 -0
  11. package/.scannerwork/scanner-report/changesets-52.pb +1 -0
  12. package/.scannerwork/scanner-report/changesets-53.pb +1 -0
  13. package/.scannerwork/scanner-report/changesets-54.pb +1 -0
  14. package/.scannerwork/scanner-report/changesets-55.pb +1 -0
  15. package/.scannerwork/scanner-report/changesets-56.pb +1 -0
  16. package/.scannerwork/scanner-report/component-1.pb +3 -2
  17. package/.scannerwork/scanner-report/component-10.pb +1 -1
  18. package/.scannerwork/scanner-report/component-11.pb +1 -1
  19. package/.scannerwork/scanner-report/component-12.pb +1 -1
  20. package/.scannerwork/scanner-report/component-13.pb +1 -1
  21. package/.scannerwork/scanner-report/component-14.pb +1 -1
  22. package/.scannerwork/scanner-report/component-15.pb +1 -1
  23. package/.scannerwork/scanner-report/component-16.pb +1 -1
  24. package/.scannerwork/scanner-report/component-17.pb +1 -1
  25. package/.scannerwork/scanner-report/component-18.pb +1 -1
  26. package/.scannerwork/scanner-report/component-19.pb +1 -1
  27. package/.scannerwork/scanner-report/component-2.pb +1 -0
  28. package/.scannerwork/scanner-report/component-20.pb +1 -1
  29. package/.scannerwork/scanner-report/component-21.pb +1 -0
  30. package/.scannerwork/scanner-report/component-22.pb +1 -1
  31. package/.scannerwork/scanner-report/component-23.pb +1 -0
  32. package/.scannerwork/scanner-report/component-24.pb +1 -0
  33. package/.scannerwork/scanner-report/component-25.pb +1 -0
  34. package/.scannerwork/scanner-report/component-37.pb +1 -1
  35. package/.scannerwork/scanner-report/component-38.pb +1 -1
  36. package/.scannerwork/scanner-report/component-39.pb +1 -1
  37. package/.scannerwork/scanner-report/component-40.pb +1 -1
  38. package/.scannerwork/scanner-report/component-41.pb +1 -1
  39. package/.scannerwork/scanner-report/component-42.pb +1 -1
  40. package/.scannerwork/scanner-report/component-43.pb +1 -1
  41. package/.scannerwork/scanner-report/component-44.pb +1 -1
  42. package/.scannerwork/scanner-report/component-49.pb +1 -0
  43. package/.scannerwork/scanner-report/component-5.pb +1 -1
  44. package/.scannerwork/scanner-report/component-50.pb +1 -0
  45. package/.scannerwork/scanner-report/component-51.pb +1 -0
  46. package/.scannerwork/scanner-report/component-52.pb +1 -0
  47. package/.scannerwork/scanner-report/component-53.pb +1 -0
  48. package/.scannerwork/scanner-report/component-54.pb +1 -0
  49. package/.scannerwork/scanner-report/component-55.pb +1 -0
  50. package/.scannerwork/scanner-report/component-56.pb +1 -0
  51. package/.scannerwork/scanner-report/component-57.pb +1 -1
  52. package/.scannerwork/scanner-report/component-58.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-61.pb +1 -1
  57. package/.scannerwork/scanner-report/component-62.pb +1 -1
  58. package/.scannerwork/scanner-report/component-67.pb +1 -1
  59. package/.scannerwork/scanner-report/component-7.pb +1 -1
  60. package/.scannerwork/scanner-report/component-8.pb +1 -1
  61. package/.scannerwork/scanner-report/component-9.pb +1 -1
  62. package/.scannerwork/scanner-report/coverages-10.pb +0 -0
  63. package/.scannerwork/scanner-report/coverages-11.pb +0 -0
  64. package/.scannerwork/scanner-report/coverages-12.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-15.pb +0 -0
  68. package/.scannerwork/scanner-report/coverages-16.pb +0 -0
  69. package/.scannerwork/scanner-report/coverages-17.pb +0 -0
  70. package/.scannerwork/scanner-report/coverages-18.pb +0 -0
  71. package/.scannerwork/scanner-report/coverages-19.pb +0 -0
  72. package/.scannerwork/scanner-report/coverages-20.pb +0 -0
  73. package/.scannerwork/scanner-report/coverages-21.pb +0 -0
  74. package/.scannerwork/scanner-report/coverages-22.pb +0 -0
  75. package/.scannerwork/scanner-report/coverages-23.pb +0 -0
  76. package/.scannerwork/scanner-report/coverages-25.pb +0 -0
  77. package/.scannerwork/scanner-report/coverages-37.pb +0 -0
  78. package/.scannerwork/scanner-report/coverages-38.pb +0 -0
  79. package/.scannerwork/scanner-report/coverages-39.pb +0 -0
  80. package/.scannerwork/scanner-report/coverages-40.pb +0 -0
  81. package/.scannerwork/scanner-report/coverages-41.pb +0 -0
  82. package/.scannerwork/scanner-report/coverages-42.pb +0 -0
  83. package/.scannerwork/scanner-report/coverages-43.pb +0 -0
  84. package/.scannerwork/scanner-report/coverages-44.pb +0 -0
  85. package/.scannerwork/scanner-report/coverages-49.pb +0 -0
  86. package/.scannerwork/scanner-report/coverages-5.pb +0 -0
  87. package/.scannerwork/scanner-report/coverages-50.pb +0 -0
  88. package/.scannerwork/scanner-report/coverages-51.pb +0 -0
  89. package/.scannerwork/scanner-report/coverages-53.pb +0 -0
  90. package/.scannerwork/scanner-report/coverages-54.pb +0 -0
  91. package/.scannerwork/scanner-report/coverages-55.pb +0 -0
  92. package/.scannerwork/scanner-report/coverages-56.pb +0 -0
  93. package/.scannerwork/scanner-report/coverages-57.pb +0 -0
  94. package/.scannerwork/scanner-report/coverages-58.pb +0 -0
  95. package/.scannerwork/scanner-report/coverages-59.pb +0 -0
  96. package/.scannerwork/scanner-report/coverages-6.pb +0 -0
  97. package/.scannerwork/scanner-report/coverages-60.pb +0 -0
  98. package/.scannerwork/scanner-report/coverages-61.pb +0 -0
  99. package/.scannerwork/scanner-report/coverages-62.pb +0 -0
  100. package/.scannerwork/scanner-report/coverages-7.pb +0 -0
  101. package/.scannerwork/scanner-report/coverages-8.pb +0 -0
  102. package/.scannerwork/scanner-report/coverages-9.pb +0 -0
  103. package/.scannerwork/scanner-report/duplications-14.pb +0 -2
  104. package/.scannerwork/scanner-report/duplications-20.pb +0 -2
  105. package/.scannerwork/scanner-report/duplications-22.pb +0 -2
  106. package/.scannerwork/scanner-report/duplications-23.pb +3 -0
  107. package/.scannerwork/scanner-report/duplications-37.pb +0 -2
  108. package/.scannerwork/scanner-report/duplications-42.pb +2 -0
  109. package/.scannerwork/scanner-report/duplications-43.pb +2 -0
  110. package/.scannerwork/scanner-report/duplications-44.pb +2 -0
  111. package/.scannerwork/scanner-report/duplications-51.pb +2 -0
  112. package/.scannerwork/scanner-report/duplications-53.pb +2 -0
  113. package/.scannerwork/scanner-report/duplications-58.pb +0 -3
  114. package/.scannerwork/scanner-report/duplications-7.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-17.pb +0 -0
  124. package/.scannerwork/scanner-report/measures-18.pb +0 -0
  125. package/.scannerwork/scanner-report/measures-19.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-25.pb +0 -0
  131. package/.scannerwork/scanner-report/measures-37.pb +0 -0
  132. package/.scannerwork/scanner-report/measures-38.pb +0 -0
  133. package/.scannerwork/scanner-report/measures-39.pb +0 -0
  134. package/.scannerwork/scanner-report/measures-40.pb +0 -0
  135. package/.scannerwork/scanner-report/measures-41.pb +0 -0
  136. package/.scannerwork/scanner-report/measures-42.pb +0 -0
  137. package/.scannerwork/scanner-report/measures-43.pb +0 -0
  138. package/.scannerwork/scanner-report/measures-44.pb +0 -0
  139. package/.scannerwork/scanner-report/measures-49.pb +0 -0
  140. package/.scannerwork/scanner-report/measures-5.pb +0 -0
  141. package/.scannerwork/scanner-report/measures-50.pb +15 -0
  142. package/.scannerwork/scanner-report/measures-51.pb +0 -0
  143. package/.scannerwork/scanner-report/measures-53.pb +0 -0
  144. package/.scannerwork/scanner-report/measures-54.pb +16 -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-57.pb +0 -0
  148. package/.scannerwork/scanner-report/measures-58.pb +0 -0
  149. package/.scannerwork/scanner-report/measures-59.pb +0 -0
  150. package/.scannerwork/scanner-report/measures-6.pb +0 -0
  151. package/.scannerwork/scanner-report/measures-60.pb +0 -0
  152. package/.scannerwork/scanner-report/measures-61.pb +0 -0
  153. package/.scannerwork/scanner-report/measures-62.pb +0 -0
  154. package/.scannerwork/scanner-report/measures-67.pb +0 -0
  155. package/.scannerwork/scanner-report/measures-7.pb +0 -0
  156. package/.scannerwork/scanner-report/measures-8.pb +0 -0
  157. package/.scannerwork/scanner-report/measures-9.pb +0 -0
  158. package/.scannerwork/scanner-report/metadata.pb +0 -0
  159. package/.scannerwork/scanner-report/source-10.txt +69 -87
  160. package/.scannerwork/scanner-report/source-11.txt +119 -249
  161. package/.scannerwork/scanner-report/source-12.txt +21 -95
  162. package/.scannerwork/scanner-report/source-13.txt +413 -142
  163. package/.scannerwork/scanner-report/source-14.txt +84 -53
  164. package/.scannerwork/scanner-report/source-15.txt +80 -92
  165. package/.scannerwork/scanner-report/source-16.txt +26 -16
  166. package/.scannerwork/scanner-report/source-17.txt +120 -115
  167. package/.scannerwork/scanner-report/source-18.txt +37 -424
  168. package/.scannerwork/scanner-report/source-19.txt +10 -95
  169. package/.scannerwork/scanner-report/{source-32.txt → source-2.txt} +49 -49
  170. package/.scannerwork/scanner-report/source-20.txt +13 -156
  171. package/.scannerwork/scanner-report/source-21.txt +40 -0
  172. package/.scannerwork/scanner-report/source-22.txt +40 -168
  173. package/.scannerwork/scanner-report/source-23.txt +213 -0
  174. package/.scannerwork/scanner-report/source-25.txt +42 -0
  175. package/.scannerwork/scanner-report/source-37.txt +47 -155
  176. package/.scannerwork/scanner-report/source-38.txt +41 -77
  177. package/.scannerwork/scanner-report/source-39.txt +266 -19
  178. package/.scannerwork/scanner-report/source-40.txt +161 -10
  179. package/.scannerwork/scanner-report/source-41.txt +97 -416
  180. package/.scannerwork/scanner-report/source-42.txt +158 -105
  181. package/.scannerwork/scanner-report/source-43.txt +148 -262
  182. package/.scannerwork/scanner-report/source-44.txt +168 -40
  183. package/.scannerwork/scanner-report/source-49.txt +59 -0
  184. package/.scannerwork/scanner-report/source-5.txt +390 -22
  185. package/.scannerwork/scanner-report/source-50.txt +113 -0
  186. package/.scannerwork/scanner-report/source-51.txt +75 -0
  187. package/.scannerwork/scanner-report/source-53.txt +92 -0
  188. package/.scannerwork/scanner-report/source-54.txt +178 -0
  189. package/.scannerwork/scanner-report/source-55.txt +129 -0
  190. package/.scannerwork/scanner-report/source-56.txt +46 -0
  191. package/.scannerwork/scanner-report/source-57.txt +251 -161
  192. package/.scannerwork/scanner-report/source-58.txt +95 -197
  193. package/.scannerwork/scanner-report/source-59.txt +25 -21
  194. package/.scannerwork/scanner-report/source-6.txt +41 -61
  195. package/.scannerwork/scanner-report/source-60.txt +104 -27
  196. package/.scannerwork/scanner-report/source-61.txt +15 -49
  197. package/.scannerwork/scanner-report/source-62.txt +65 -85
  198. package/.scannerwork/scanner-report/source-67.txt +424 -75
  199. package/.scannerwork/scanner-report/source-7.txt +57 -68
  200. package/.scannerwork/scanner-report/source-8.txt +98 -43
  201. package/.scannerwork/scanner-report/source-9.txt +167 -21
  202. package/.scannerwork/scanner-report/symbols-10.pb +69 -30
  203. package/.scannerwork/scanner-report/symbols-11.pb +91 -352
  204. package/.scannerwork/scanner-report/symbols-12.pb +13 -59
  205. package/.scannerwork/scanner-report/symbols-13.pb +785 -139
  206. package/.scannerwork/scanner-report/symbols-14.pb +59 -42
  207. package/.scannerwork/scanner-report/symbols-15.pb +57 -45
  208. package/.scannerwork/scanner-report/symbols-16.pb +31 -14
  209. package/.scannerwork/scanner-report/symbols-17.pb +71 -33
  210. package/.scannerwork/scanner-report/symbols-18.pb +9 -0
  211. package/.scannerwork/scanner-report/symbols-19.pb +8 -28
  212. package/.scannerwork/scanner-report/symbols-20.pb +9 -153
  213. package/.scannerwork/scanner-report/symbols-21.pb +23 -0
  214. package/.scannerwork/scanner-report/symbols-22.pb +24 -180
  215. package/.scannerwork/scanner-report/symbols-23.pb +264 -0
  216. package/.scannerwork/scanner-report/symbols-25.pb +19 -0
  217. package/.scannerwork/scanner-report/symbols-37.pb +38 -202
  218. package/.scannerwork/scanner-report/symbols-38.pb +32 -57
  219. package/.scannerwork/scanner-report/symbols-39.pb +421 -19
  220. package/.scannerwork/scanner-report/symbols-40.pb +207 -21
  221. package/.scannerwork/scanner-report/symbols-41.pb +85 -789
  222. package/.scannerwork/scanner-report/symbols-42.pb +203 -86
  223. package/.scannerwork/scanner-report/symbols-43.pb +152 -420
  224. package/.scannerwork/scanner-report/symbols-44.pb +180 -24
  225. package/.scannerwork/scanner-report/symbols-49.pb +19 -0
  226. package/.scannerwork/scanner-report/symbols-5.pb +604 -20
  227. package/.scannerwork/scanner-report/symbols-50.pb +46 -0
  228. package/.scannerwork/scanner-report/symbols-51.pb +42 -0
  229. package/.scannerwork/scanner-report/symbols-53.pb +44 -0
  230. package/.scannerwork/scanner-report/symbols-54.pb +144 -0
  231. package/.scannerwork/scanner-report/symbols-55.pb +33 -0
  232. package/.scannerwork/scanner-report/symbols-56.pb +20 -0
  233. package/.scannerwork/scanner-report/symbols-57.pb +336 -191
  234. package/.scannerwork/scanner-report/symbols-58.pb +59 -264
  235. package/.scannerwork/scanner-report/symbols-59.pb +18 -9
  236. package/.scannerwork/scanner-report/symbols-6.pb +21 -46
  237. package/.scannerwork/scanner-report/symbols-60.pb +30 -31
  238. package/.scannerwork/scanner-report/symbols-61.pb +13 -32
  239. package/.scannerwork/scanner-report/symbols-62.pb +45 -68
  240. package/.scannerwork/scanner-report/symbols-7.pb +32 -44
  241. package/.scannerwork/scanner-report/symbols-8.pb +28 -19
  242. package/.scannerwork/scanner-report/symbols-9.pb +226 -17
  243. package/.scannerwork/scanner-report/syntax-highlightings-10.pb +76 -82
  244. package/.scannerwork/scanner-report/syntax-highlightings-11.pb +157 -205
  245. package/.scannerwork/scanner-report/syntax-highlightings-12.pb +27 -98
  246. package/.scannerwork/scanner-report/syntax-highlightings-13.pb +535 -153
  247. package/.scannerwork/scanner-report/syntax-highlightings-14.pb +144 -44
  248. package/.scannerwork/scanner-report/syntax-highlightings-15.pb +122 -94
  249. package/.scannerwork/scanner-report/syntax-highlightings-16.pb +39 -15
  250. package/.scannerwork/scanner-report/syntax-highlightings-17.pb +141 -89
  251. package/.scannerwork/scanner-report/syntax-highlightings-18.pb +50 -3330
  252. package/.scannerwork/scanner-report/syntax-highlightings-19.pb +16 -105
  253. package/.scannerwork/scanner-report/{syntax-highlightings-32.pb → syntax-highlightings-2.pb} +26 -26
  254. package/.scannerwork/scanner-report/syntax-highlightings-20.pb +23 -183
  255. package/.scannerwork/scanner-report/syntax-highlightings-21.pb +56 -0
  256. package/.scannerwork/scanner-report/syntax-highlightings-22.pb +66 -194
  257. package/.scannerwork/scanner-report/syntax-highlightings-23.pb +296 -0
  258. package/.scannerwork/scanner-report/syntax-highlightings-25.pb +66 -0
  259. package/.scannerwork/scanner-report/syntax-highlightings-37.pb +55 -234
  260. package/.scannerwork/scanner-report/syntax-highlightings-38.pb +68 -112
  261. package/.scannerwork/scanner-report/syntax-highlightings-39.pb +668 -26
  262. package/.scannerwork/scanner-report/syntax-highlightings-40.pb +250 -16
  263. package/.scannerwork/scanner-report/syntax-highlightings-41.pb +89 -589
  264. package/.scannerwork/scanner-report/syntax-highlightings-42.pb +243 -88
  265. package/.scannerwork/scanner-report/syntax-highlightings-43.pb +169 -663
  266. package/.scannerwork/scanner-report/syntax-highlightings-44.pb +194 -66
  267. package/.scannerwork/scanner-report/syntax-highlightings-49.pb +59 -0
  268. package/.scannerwork/scanner-report/syntax-highlightings-5.pb +610 -33
  269. package/.scannerwork/scanner-report/syntax-highlightings-50.pb +126 -0
  270. package/.scannerwork/scanner-report/syntax-highlightings-51.pb +78 -0
  271. package/.scannerwork/scanner-report/syntax-highlightings-53.pb +90 -0
  272. package/.scannerwork/scanner-report/syntax-highlightings-54.pb +243 -0
  273. package/.scannerwork/scanner-report/syntax-highlightings-55.pb +129 -0
  274. package/.scannerwork/scanner-report/syntax-highlightings-56.pb +64 -0
  275. package/.scannerwork/scanner-report/syntax-highlightings-57.pb +180 -185
  276. package/.scannerwork/scanner-report/syntax-highlightings-58.pb +93 -256
  277. package/.scannerwork/scanner-report/syntax-highlightings-59.pb +21 -32
  278. package/.scannerwork/scanner-report/syntax-highlightings-6.pb +59 -74
  279. package/.scannerwork/scanner-report/syntax-highlightings-60.pb +83 -39
  280. package/.scannerwork/scanner-report/syntax-highlightings-61.pb +15 -80
  281. package/.scannerwork/scanner-report/syntax-highlightings-62.pb +61 -73
  282. package/.scannerwork/scanner-report/syntax-highlightings-67.pb +3331 -90
  283. package/.scannerwork/scanner-report/syntax-highlightings-7.pb +75 -67
  284. package/.scannerwork/scanner-report/syntax-highlightings-8.pb +108 -29
  285. package/.scannerwork/scanner-report/syntax-highlightings-9.pb +339 -21
  286. package/CHANGES.md +1 -1
  287. package/package.json +1 -1
  288. package/test-18-results.xml +159 -159
  289. package/test-iam-18-results.xml +49 -49
  290. package/.scannerwork/scanner-report/changesets-29.pb +0 -1
  291. package/.scannerwork/scanner-report/changesets-30.pb +0 -1
  292. package/.scannerwork/scanner-report/changesets-34.pb +0 -1
  293. package/.scannerwork/scanner-report/changesets-35.pb +0 -1
  294. package/.scannerwork/scanner-report/changesets-36.pb +0 -1
  295. package/.scannerwork/scanner-report/changesets-4.pb +0 -1
  296. package/.scannerwork/scanner-report/changesets-45.pb +0 -1
  297. package/.scannerwork/scanner-report/changesets-63.pb +0 -1
  298. package/.scannerwork/scanner-report/changesets-64.pb +0 -1
  299. package/.scannerwork/scanner-report/changesets-65.pb +0 -1
  300. package/.scannerwork/scanner-report/changesets-66.pb +0 -1
  301. package/.scannerwork/scanner-report/changesets-67.pb +0 -1
  302. package/.scannerwork/scanner-report/changesets-68.pb +0 -1
  303. package/.scannerwork/scanner-report/component-29.pb +0 -1
  304. package/.scannerwork/scanner-report/component-30.pb +0 -1
  305. package/.scannerwork/scanner-report/component-32.pb +0 -1
  306. package/.scannerwork/scanner-report/component-34.pb +0 -1
  307. package/.scannerwork/scanner-report/component-35.pb +0 -1
  308. package/.scannerwork/scanner-report/component-36.pb +0 -1
  309. package/.scannerwork/scanner-report/component-4.pb +0 -1
  310. package/.scannerwork/scanner-report/component-45.pb +0 -1
  311. package/.scannerwork/scanner-report/component-63.pb +0 -1
  312. package/.scannerwork/scanner-report/component-64.pb +0 -1
  313. package/.scannerwork/scanner-report/component-65.pb +0 -1
  314. package/.scannerwork/scanner-report/component-66.pb +0 -1
  315. package/.scannerwork/scanner-report/component-68.pb +0 -1
  316. package/.scannerwork/scanner-report/coverages-29.pb +0 -0
  317. package/.scannerwork/scanner-report/coverages-30.pb +0 -0
  318. package/.scannerwork/scanner-report/coverages-34.pb +0 -0
  319. package/.scannerwork/scanner-report/coverages-35.pb +0 -0
  320. package/.scannerwork/scanner-report/coverages-36.pb +0 -0
  321. package/.scannerwork/scanner-report/coverages-45.pb +0 -0
  322. package/.scannerwork/scanner-report/coverages-63.pb +0 -0
  323. package/.scannerwork/scanner-report/coverages-64.pb +0 -0
  324. package/.scannerwork/scanner-report/coverages-65.pb +0 -0
  325. package/.scannerwork/scanner-report/coverages-67.pb +0 -0
  326. package/.scannerwork/scanner-report/coverages-68.pb +0 -0
  327. package/.scannerwork/scanner-report/duplications-64.pb +0 -2
  328. package/.scannerwork/scanner-report/duplications-66.pb +0 -0
  329. package/.scannerwork/scanner-report/duplications-67.pb +0 -0
  330. package/.scannerwork/scanner-report/measures-29.pb +0 -0
  331. package/.scannerwork/scanner-report/measures-30.pb +0 -0
  332. package/.scannerwork/scanner-report/measures-34.pb +0 -0
  333. package/.scannerwork/scanner-report/measures-35.pb +0 -0
  334. package/.scannerwork/scanner-report/measures-36.pb +0 -0
  335. package/.scannerwork/scanner-report/measures-45.pb +0 -0
  336. package/.scannerwork/scanner-report/measures-63.pb +0 -15
  337. package/.scannerwork/scanner-report/measures-64.pb +0 -0
  338. package/.scannerwork/scanner-report/measures-65.pb +0 -0
  339. package/.scannerwork/scanner-report/measures-68.pb +0 -0
  340. package/.scannerwork/scanner-report/source-29.txt +0 -81
  341. package/.scannerwork/scanner-report/source-30.txt +0 -60
  342. package/.scannerwork/scanner-report/source-34.txt +0 -414
  343. package/.scannerwork/scanner-report/source-35.txt +0 -134
  344. package/.scannerwork/scanner-report/source-36.txt +0 -106
  345. package/.scannerwork/scanner-report/source-45.txt +0 -37
  346. package/.scannerwork/scanner-report/source-63.txt +0 -151
  347. package/.scannerwork/scanner-report/source-64.txt +0 -187
  348. package/.scannerwork/scanner-report/source-65.txt +0 -32
  349. package/.scannerwork/scanner-report/source-68.txt +0 -29
  350. package/.scannerwork/scanner-report/symbols-29.pb +0 -32
  351. package/.scannerwork/scanner-report/symbols-30.pb +0 -21
  352. package/.scannerwork/scanner-report/symbols-34.pb +0 -604
  353. package/.scannerwork/scanner-report/symbols-35.pb +0 -71
  354. package/.scannerwork/scanner-report/symbols-36.pb +0 -59
  355. package/.scannerwork/scanner-report/symbols-45.pb +0 -13
  356. package/.scannerwork/scanner-report/symbols-63.pb +0 -93
  357. package/.scannerwork/scanner-report/symbols-64.pb +0 -227
  358. package/.scannerwork/scanner-report/symbols-65.pb +0 -9
  359. package/.scannerwork/scanner-report/symbols-67.pb +0 -39
  360. package/.scannerwork/scanner-report/symbols-68.pb +0 -8
  361. package/.scannerwork/scanner-report/syntax-highlightings-29.pb +0 -98
  362. package/.scannerwork/scanner-report/syntax-highlightings-30.pb +0 -80
  363. package/.scannerwork/scanner-report/syntax-highlightings-34.pb +0 -641
  364. package/.scannerwork/scanner-report/syntax-highlightings-35.pb +0 -181
  365. package/.scannerwork/scanner-report/syntax-highlightings-36.pb +0 -178
  366. package/.scannerwork/scanner-report/syntax-highlightings-45.pb +0 -62
  367. package/.scannerwork/scanner-report/syntax-highlightings-63.pb +0 -237
  368. package/.scannerwork/scanner-report/syntax-highlightings-64.pb +0 -368
  369. package/.scannerwork/scanner-report/syntax-highlightings-65.pb +0 -54
  370. package/.scannerwork/scanner-report/syntax-highlightings-68.pb +0 -49
  371. /package/.scannerwork/scanner-report/{coverages-66.pb → coverages-24.pb} +0 -0
  372. /package/.scannerwork/scanner-report/{coverages-4.pb → coverages-52.pb} +0 -0
  373. /package/.scannerwork/scanner-report/{duplications-19.pb → duplications-18.pb} +0 -0
  374. /package/.scannerwork/scanner-report/{duplications-29.pb → duplications-21.pb} +0 -0
  375. /package/.scannerwork/scanner-report/{duplications-30.pb → duplications-24.pb} +0 -0
  376. /package/.scannerwork/scanner-report/{duplications-34.pb → duplications-25.pb} +0 -0
  377. /package/.scannerwork/scanner-report/{duplications-35.pb → duplications-49.pb} +0 -0
  378. /package/.scannerwork/scanner-report/{duplications-36.pb → duplications-50.pb} +0 -0
  379. /package/.scannerwork/scanner-report/{duplications-4.pb → duplications-52.pb} +0 -0
  380. /package/.scannerwork/scanner-report/{duplications-45.pb → duplications-54.pb} +0 -0
  381. /package/.scannerwork/scanner-report/{duplications-63.pb → duplications-55.pb} +0 -0
  382. /package/.scannerwork/scanner-report/{duplications-65.pb → duplications-56.pb} +0 -0
  383. /package/.scannerwork/scanner-report/{issues-58.pb → issues-23.pb} +0 -0
  384. /package/.scannerwork/scanner-report/{issues-66.pb → issues-24.pb} +0 -0
  385. /package/.scannerwork/scanner-report/{issues-39.pb → issues-25.pb} +0 -0
  386. /package/.scannerwork/scanner-report/{issues-42.pb → issues-41.pb} +0 -0
  387. /package/.scannerwork/scanner-report/{measures-32.pb → measures-2.pb} +0 -0
  388. /package/.scannerwork/scanner-report/{measures-66.pb → measures-24.pb} +0 -0
  389. /package/.scannerwork/scanner-report/{measures-4.pb → measures-52.pb} +0 -0
  390. /package/.scannerwork/scanner-report/{source-66.txt → source-24.txt} +0 -0
  391. /package/.scannerwork/scanner-report/{source-4.txt → source-52.txt} +0 -0
  392. /package/.scannerwork/scanner-report/{symbols-66.pb → symbols-24.pb} +0 -0
  393. /package/.scannerwork/scanner-report/{symbols-4.pb → symbols-52.pb} +0 -0
  394. /package/.scannerwork/scanner-report/{syntax-highlightings-66.pb → syntax-highlightings-24.pb} +0 -0
  395. /package/.scannerwork/scanner-report/{syntax-highlightings-4.pb → syntax-highlightings-52.pb} +0 -0
@@ -1,4 +1,4 @@
1
- // Copyright © 2017, 2023 IBM Corp. All rights reserved.
1
+ // Copyright © 2017, 2018 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.
@@ -12,278 +12,164 @@
12
12
  // See the License for the specific language governing permissions and
13
13
  // limitations under the License.
14
14
 
15
- /* global describe it before after beforeEach */
16
- 'use strict';
17
-
18
- const assert = require('assert');
19
- const fs = require('fs');
20
- const u = require('./citestutils.js');
21
- const mockServerPort = +process.env.COUCHBACKUP_MOCK_SERVER_PORT || 7777;
22
- const { once } = require('node:events');
23
- const url = `http://localhost:${mockServerPort}`;
24
- const nock = require('nock');
25
- const httpProxy = require('http-proxy');
26
- const Readable = require('stream').Readable;
27
-
28
- // Create an infinite stream to read.
29
- // It just keeps sending a backup line, useful for testing cases of
30
- // termination while a stream has content remaining (the animaldb backup
31
- // is too small for that).
32
- class InfiniteBackupStream extends Readable {
33
- constructor(opt) {
34
- super(opt);
35
- this.contents = Buffer.from('[{"_id":"giraffe","_rev":"3-7665c3e66315ff40616cceef62886bd8","min_weight":830,"min_length":5,"max_weight":1600,"max_length":6,"wiki_page":"http://en.wikipedia.org/wiki/Giraffe","class":"mammal","diet":"herbivore","_revisions":{"start":3,"ids":["7665c3e66315ff40616cceef62886bd8","aaaf10d5a68cdf22d95a5482a0e95549","967a00dff5e02add41819138abb3284d"]}}]\n', 'utf8');
36
- }
15
+ // Small script which backs up a Cloudant or CouchDB database to an S3
16
+ // bucket via a stream rather than on-disk file.
17
+ //
18
+ // The script generates the backup object name by combining together the path
19
+ // part of the database URL and the current time.
37
20
 
38
- _read() {
39
- let proceed;
40
- do {
41
- proceed = this.push(this.contents);
42
- } while (proceed);
43
- }
44
- }
21
+ 'use strict';
45
22
 
46
- function assertNock() {
47
- try {
48
- assert.ok(nock.isDone());
49
- } catch (err) {
50
- console.error('pending mocks: %j', nock.pendingMocks());
51
- throw err;
23
+ const stream = require('stream');
24
+ const url = require('url');
25
+
26
+ const AWS = require('aws-sdk');
27
+ const couchbackup = require('@cloudant/couchbackup');
28
+ const debug = require('debug')('s3-backup');
29
+ const VError = require('verror').VError;
30
+
31
+ /*
32
+ Main function, run from base of file.
33
+ */
34
+ function main() {
35
+ const argv = require('yargs')
36
+ .usage('Usage: $0 [options]')
37
+ .example('$0 -s https://user:pass@host/db -b <bucket>', 'Backup db to bucket')
38
+ .options({
39
+ source: { alias: 's', nargs: 1, demandOption: true, describe: 'Source database URL' },
40
+ bucket: { alias: 'b', nargs: 1, demandOption: true, describe: 'Destination bucket' },
41
+ prefix: { alias: 'p', nargs: 1, describe: 'Prefix for backup object key', default: 'couchbackup' },
42
+ s3url: { nargs: 1, describe: 'S3 endpoint URL' },
43
+ awsprofile: { nargs: 1, describe: 'The profile section to use in the ~/.aws/credentials file', default: 'default' }
44
+ })
45
+ .help('h').alias('h', 'help')
46
+ .epilog('Copyright (C) IBM 2017')
47
+ .argv;
48
+
49
+ const sourceUrl = argv.source;
50
+ const backupName = new url.URL(sourceUrl).pathname.split('/').filter(function(x) { return x; }).join('-');
51
+ const backupBucket = argv.bucket;
52
+ const backupKeyPrefix = `${argv.prefix}-${backupName}`;
53
+ const shallow = argv.shallow;
54
+
55
+ const backupKey = `${backupKeyPrefix}-${new Date().toISOString()}`;
56
+
57
+ const s3Endpoint = argv.s3url;
58
+ const awsProfile = argv.awsprofile;
59
+
60
+ // Creds are from ~/.aws/credentials, environment etc. (see S3 docs).
61
+ const awsOpts = {
62
+ signatureVersion: 'v4',
63
+ credentials: new AWS.SharedIniFileCredentials({ profile: awsProfile })
64
+ };
65
+ if (typeof s3Endpoint !== 'undefined') {
66
+ awsOpts.endpoint = new AWS.Endpoint(s3Endpoint);
52
67
  }
53
- }
54
-
55
- function testPromiseWithAssertNock(testPromise) {
56
- return testPromise.finally(() => {
57
- assertNock();
58
- });
59
- }
60
-
61
- async function backupHttpError(opts, errorName, errorCode) {
62
- const p = u.p(opts, { expectedBackupError: { name: errorName, code: errorCode } });
68
+ const s3 = new AWS.S3(awsOpts);
63
69
 
64
- // Create a file and attempt a backup to it
65
- const output = fs.createWriteStream('/dev/null');
66
- return once(output, 'open')
70
+ debug(`Creating a new backup of ${s(sourceUrl)} at ${backupBucket}/${backupKey}...`);
71
+ bucketAccessible(s3, backupBucket)
67
72
  .then(() => {
68
- return testPromiseWithAssertNock(u.testBackup(p, 'fakenockdb', output));
73
+ return backupToS3(sourceUrl, s3, backupBucket, backupKey, shallow);
74
+ })
75
+ .then(() => {
76
+ debug('done.');
77
+ })
78
+ .catch((reason) => {
79
+ debug(`Error: ${reason}`);
80
+ process.exit(1);
69
81
  });
70
82
  }
71
83
 
72
- async function restoreHttpError(opts, errorName, errorCode) {
73
- const q = u.p(opts, { expectedRestoreError: { name: errorName, code: errorCode } });
74
- return testPromiseWithAssertNock(u.testRestoreFromFile(q, './test/fixtures/animaldb_expected.json', 'fakenockdb'));
75
- }
76
-
77
- [{ useApi: true }, { useApi: false }].forEach(function(params) {
78
- describe(u.scenario('#unit Fatal errors', params), function() {
79
- let processEnvCopy;
80
- let proxy;
81
-
82
- before('Set process data for test', function() {
83
- const proxyPort = mockServerPort + 1000;
84
- // Copy env and argv so we can reset them after the tests
85
- processEnvCopy = JSON.parse(JSON.stringify(process.env));
86
-
87
- // Set up a proxy to point to our nock server because the nock override
88
- // isn't visible to the spawned CLI process
89
- if (!params.useApi) {
90
- proxy = httpProxy.createProxyServer({ target: url }).listen(proxyPort, 'localhost');
91
- proxy.on('error', (err, req, res) => {
92
- console.log(`Proxy received error ${err}`);
93
- res.writeHead(400, {
94
- 'Content-Type': 'application/json'
95
- });
96
- res.end(JSON.stringify(err));
97
- });
98
- }
99
-
100
- // setup environment variables
101
- process.env.COUCH_URL = (params.useApi) ? url : `http://localhost:${proxyPort}`;
102
-
103
- nock.emitter.on('no match', (req, opts) => {
104
- console.error(`Unmatched nock request ${opts.method} ${opts.protocol}${opts.host}${opts.path}`);
105
- });
106
- });
107
-
108
- after('Reset process data', function(done) {
109
- process.env = processEnvCopy;
110
- nock.emitter.removeAllListeners();
111
- if (!params.useApi) {
112
- proxy.close(done);
84
+ /**
85
+ * Return a promise that resolves if the bucket is available and
86
+ * rejects if not.
87
+ *
88
+ * @param {any} s3 S3 client object
89
+ * @param {any} bucketName Bucket name
90
+ * @returns Promise
91
+ */
92
+ function bucketAccessible(s3, bucketName) {
93
+ return new Promise(function(resolve, reject) {
94
+ const params = {
95
+ Bucket: bucketName
96
+ };
97
+ s3.headBucket(params, function(err, data) {
98
+ if (err) {
99
+ reject(new VError(err, 'S3 bucket not accessible'));
113
100
  } else {
114
- done();
101
+ resolve();
115
102
  }
116
103
  });
104
+ });
105
+ }
117
106
 
118
- beforeEach('Reset nocks', function() {
119
- nock.cleanAll();
120
- });
121
-
122
- describe('for backup', function() {
123
- it('should terminate when DB does not exist', function() {
124
- // Simulate existence check
125
- nock(url).head('/fakenockdb').reply(404, { error: 'not_found', reason: 'missing' });
126
- return backupHttpError(params, 'DatabaseNotFound', 10);
127
- });
128
-
129
- it('should terminate on BulkGetError', function() {
130
- // Simulate existence check
131
- const n = nock(url).head('/fakenockdb').reply(200);
132
- // Simulate _bulk_get not available
133
- n.post('/fakenockdb/_bulk_get').reply(404, { error: 'not_found', reason: 'missing' });
134
- return backupHttpError(params, 'BulkGetError', 50);
135
- });
136
-
137
- it('should terminate on Unauthorized existence check', function() {
138
- // Simulate a 401
139
- nock(url).head('/fakenockdb').reply(401, { error: 'unauthorized', reason: '_reader access is required for this request' });
140
- return backupHttpError(params, 'Unauthorized', 11);
141
- });
142
-
143
- it('should terminate on Forbidden no _reader', function() {
144
- // Simulate a 403
145
- nock(url).head('/fakenockdb').reply(403, { error: 'forbidden', reason: '_reader access is required for this request' });
146
- return backupHttpError(params, 'Forbidden', 12);
147
- });
148
-
149
- it('should terminate on _bulk_get HTTPFatalError', function() {
150
- // Provide a mock complete changes log to allow a resume to skip ahead
151
- const p = u.p(params, { opts: { resume: true, log: './test/fixtures/test.log' } });
152
- // Allow the existence and _bulk_get checks to pass
153
- const n = nock(url).head('/fakenockdb').reply(200);
154
- n.post('/fakenockdb/_bulk_get').reply(200, '{"results": []}');
155
- // Simulate a fatal HTTP error when trying to fetch docs
156
- // Note: 2 outstanding batches, so 2 responses, 1 mock is optional because we can't guarantee timing
157
- n.post('/fakenockdb/_bulk_get').query(true).reply(400, { error: 'bad_request', reason: 'testing bad response' });
158
- n.post('/fakenockdb/_bulk_get').query(true).optionally().reply(400, { error: 'bad_request', reason: 'testing bad response' });
159
- return backupHttpError(p, 'HTTPFatalError', 40);
160
- });
161
-
162
- it('should terminate on NoLogFileName', function() {
163
- // Don't supply a log file name with resume
164
- const p = u.p(params, { opts: { resume: true } });
165
- return backupHttpError(p, 'NoLogFileName', 20);
166
- });
167
-
168
- it('should terminate on LogDoesNotExist', function() {
169
- // Use a non-existent log file
170
- const p = u.p(params, { opts: { resume: true, log: './test/fixtures/doesnotexist.log' } });
171
- return backupHttpError(p, 'LogDoesNotExist', 21);
172
- });
173
-
174
- it('should terminate on IncompleteChangesInLogFile', function() {
175
- // Use an incomplete changes log file
176
- const p = u.p(params, { opts: { resume: true, log: './test/fixtures/incomplete_changes.log' } });
177
- // Allow the existence and _bulk_get checks to pass
178
- const n = nock(url).head('/fakenockdb').reply(200);
179
- n.post('/fakenockdb/_bulk_get').reply(200, '{"results": []}');
180
- // Should fail when it reads the incomplete changes
181
- return backupHttpError(p, 'IncompleteChangesInLogFile', 22);
182
- });
183
-
184
- it('should terminate on _changes HTTPFatalError', function() {
185
- // Allow the existence and _bulk_get checks to pass
186
- const n = nock(url).head('/fakenockdb').reply(200);
187
- n.post('/fakenockdb/_bulk_get').reply(200, '{"results": []}');
188
- // Simulate a fatal HTTP error when trying to fetch docs (note 2 outstanding batches)
189
- n.post('/fakenockdb/_changes').query(true).reply(400, { error: 'bad_request', reason: 'testing bad response' });
190
- return backupHttpError(params, 'HTTPFatalError', 40);
191
- });
192
-
193
- it('should terminate on SpoolChangesError', function() {
194
- // Allow the existence and _bulk_get checks to pass
195
- const n = nock(url).head('/fakenockdb').reply(200);
196
- n.post('/fakenockdb/_bulk_get').reply(200, '{"results": []}');
197
- // Simulate a changes without a last_seq
198
- n.post('/fakenockdb/_changes').query(true).reply(200,
199
- {
200
- results: [{
201
- seq: '2-g1AAAAEbeJzLYWBgYMlgTmFQSElKzi9KdUhJstTLTS3KLElMT9VLzskvTUnMK9HLSy3JAapkSmRIsv___39WBnMiUy5QgN3MzDIxOdEMWb85dv0gSxThigyN8diS5AAkk-pBFiUyoOkzxKMvjwVIMjQAKaDW_Zh6TQnqPQDRC7I3CwDPDV1k',
202
- id: 'badger',
203
- changes: [{ rev: '4-51aa94e4b0ef37271082033bba52b850' }]
204
- }]
205
- });
206
- return backupHttpError(params, 'SpoolChangesError', 30);
207
- });
208
- });
209
-
210
- describe('for restore', function() {
211
- it('should terminate on Unauthorized db existence check', function() {
212
- // Simulate a 401
213
- nock(url).get('/fakenockdb').reply(401, { error: 'unauthorized', reason: '_reader access is required for this request' });
214
- return restoreHttpError(params, 'Unauthorized', 11);
215
- });
216
-
217
- it('should terminate on Forbidden no _writer', function() {
218
- // Simulate the DB exists (i.e. you can read it)
219
- const n = nock(url).get('/fakenockdb').reply(200, { doc_count: 0, doc_del_count: 0 });
220
- // Simulate a 403 trying to write
221
- n.post('/fakenockdb/_bulk_docs').reply(403, { error: 'forbidden', reason: '_writer access is required for this request' });
222
- return restoreHttpError(params, 'Forbidden', 12);
223
- });
224
-
225
- it('should terminate on RestoreDatabaseNotFound', function() {
226
- // Simulate the DB does not exist
227
- nock(url).get('/fakenockdb').reply(404, { error: 'not_found', reason: 'Database does not exist.' });
228
- return restoreHttpError(params, 'DatabaseNotFound', 10);
229
- });
230
-
231
- it('should terminate on notEmptyDBErr when database is not empty', function() {
232
- // Simulate the DB that does exist and not empty
233
- nock(url).get('/fakenockdb').reply(200, { doc_count: 10, doc_del_count: 0 });
234
- return restoreHttpError(params, 'DatabaseNotEmpty', 13);
235
- });
236
-
237
- it('should terminate on notEmptyDBErr when database is not new', function() {
238
- // Simulate the DB that does exist and not new
239
- nock(url).get('/fakenockdb').reply(200, { doc_count: 0, doc_del_count: 10 });
240
- return restoreHttpError(params, 'DatabaseNotEmpty', 13);
241
- });
242
-
243
- it('should terminate on _bulk_docs HTTPFatalError', function() {
244
- // Simulate the DB exists
245
- const n = nock(url).get('/fakenockdb').reply(200, { doc_count: 0, doc_del_count: 0 });
246
- // Use a parallelism of one and mock one response
247
- const p = u.p(params, { opts: { parallelism: 1 } });
248
- // Simulate a 400 trying to write
249
- n.post('/fakenockdb/_bulk_docs').reply(400, { error: 'bad_request', reason: 'testing bad response' });
250
- return restoreHttpError(p, 'HTTPFatalError', 40);
251
- });
252
-
253
- it('should terminate on _bulk_docs HTTPFatalError from system database', function() {
254
- // Simulate that target database exists and is _not_ empty.
255
- // This should pass validator as we exclude system databases from the check.
256
- const n = nock(url).get('/_replicator').reply(200, { doc_count: 1, doc_del_count: 0 });
257
- // Simulate a 400 trying to write
258
- n.post('/_replicator/_bulk_docs').reply(400, { error: 'bad_request', reason: 'testing bad response' });
259
- // Use a parallelism of one and mock one response
260
- const q = u.p(params, { opts: { parallelism: 1 }, expectedRestoreError: { name: 'HTTPFatalError', code: 40 } });
261
- return testPromiseWithAssertNock(u.testRestore(q, new InfiniteBackupStream(), '_replicator'));
262
- });
107
+ /**
108
+ * Backup directly from Cloudant to an object store object via a stream.
109
+ *
110
+ * @param {any} sourceUrl URL of database
111
+ * @param {any} s3Client Object store client
112
+ * @param {any} s3Bucket Backup destination bucket
113
+ * @param {any} s3Key Backup destination key name (shouldn't exist)
114
+ * @param {any} shallow Whether to use the couchbackup `shallow` mode
115
+ * @returns Promise
116
+ */
117
+ function backupToS3(sourceUrl, s3Client, s3Bucket, s3Key, shallow) {
118
+ return new Promise((resolve, reject) => {
119
+ debug(`Setting up S3 upload to ${s3Bucket}/${s3Key}`);
120
+
121
+ // A pass through stream that has couchbackup's output
122
+ // written to it and it then read by the S3 upload client.
123
+ // It has a 64MB highwater mark to allow for fairly
124
+ // uneven network connectivity.
125
+ const streamToUpload = new stream.PassThrough({ highWaterMark: 67108864 });
126
+
127
+ // Set up S3 upload.
128
+ const params = {
129
+ Bucket: s3Bucket,
130
+ Key: s3Key,
131
+ Body: streamToUpload
132
+ };
133
+ s3Client.upload(params, function(err, data) {
134
+ debug('Object store upload done');
135
+ if (err) {
136
+ debug(err);
137
+ reject(new VError(err, 'Object store upload failed'));
138
+ return;
139
+ }
140
+ debug('Object store upload succeeded');
141
+ debug(data);
142
+ resolve();
143
+ }).httpUploadProgress = (progress) => {
144
+ debug(`Object store upload progress: ${progress}`);
145
+ };
146
+
147
+ debug(`Starting streaming data from ${s(sourceUrl)}`);
148
+ couchbackup.backup(
149
+ sourceUrl,
150
+ streamToUpload,
151
+ (err, obj) => {
152
+ if (err) {
153
+ debug(err);
154
+ reject(new VError(err, 'CouchBackup failed with an error'));
155
+ return;
156
+ }
157
+ debug(`Download from ${s(sourceUrl)} complete.`);
158
+ streamToUpload.end(); // must call end() to complete upload.
159
+ // resolve() is called by the upload
160
+ }
161
+ );
162
+ });
163
+ }
263
164
 
264
- it('should terminate on _bulk_docs HTTPFatalError large stream', function() {
265
- // Simulate the DB exists
266
- const n = nock(url).get('/fakenockdb').reply(200, { doc_count: 0, doc_del_count: 0 });
267
- // Simulate a 400 trying to write
268
- // Provide a body function to handle the stream, but allow any body
269
- n.post('/fakenockdb/_bulk_docs', function(body) { return true; }).reply(400, { error: 'bad_request', reason: 'testing bad response' });
270
- // Use only parallelism 1 so we don't have to mock up loads of responses
271
- const q = u.p(params, { opts: { parallelism: 1 }, expectedRestoreError: { name: 'HTTPFatalError', code: 40 } });
272
- return testPromiseWithAssertNock(u.testRestore(q, new InfiniteBackupStream(), 'fakenockdb'));
273
- });
165
+ /**
166
+ * Remove creds from a URL, e.g., before logging
167
+ *
168
+ * @param {string} url URL to safen
169
+ */
170
+ function s(originalUrl) {
171
+ const parts = new url.URL(originalUrl);
172
+ return url.format(parts, { auth: false });
173
+ }
274
174
 
275
- it('should terminate on multiple _bulk_docs HTTPFatalError', function() {
276
- // Simulate the DB exists
277
- const n = nock(url).get('/fakenockdb').reply(200, { doc_count: 0, doc_del_count: 0 });
278
- // Simulate a 400 trying to write docs, 5 times because of default parallelism
279
- // Provide a body function to handle the stream, but allow any body
280
- // Four of the mocks are optional because of parallelism 5 we can't guarantee that the exit will happen
281
- // after all 5 requests, but we must get at least one of them
282
- n.post('/fakenockdb/_bulk_docs', function(body) { return true; }).reply(400, { error: 'bad_request', reason: 'testing bad response' });
283
- n.post('/fakenockdb/_bulk_docs', function(body) { return true; }).times(4).optionally().reply(400, { error: 'bad_request', reason: 'testing bad response' });
284
- const q = u.p(params, { opts: { bufferSize: 1 }, expectedRestoreError: { name: 'HTTPFatalError', code: 40 } });
285
- return restoreHttpError(q, 'HTTPFatalError', 40);
286
- });
287
- });
288
- });
289
- });
175
+ main();
@@ -1,4 +1,4 @@
1
- // Copyright © 2017, 2023 IBM Corp. All rights reserved.
1
+ // Copyright © 2017, 2018 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.
@@ -12,51 +12,179 @@
12
12
  // See the License for the specific language governing permissions and
13
13
  // limitations under the License.
14
14
 
15
- /* global describe it */
15
+ // Small script which backs up a Cloudant or CouchDB database to an S3
16
+ // bucket, using an intermediary file on disk.
17
+ //
18
+ // The script generates the backup object name by combining together the path
19
+ // part of the database URL and the current time.
20
+
16
21
  'use strict';
17
22
 
23
+ const stream = require('stream');
18
24
  const fs = require('fs');
19
- const { once } = require('node:events');
20
- const u = require('./citestutils.js');
21
-
22
- [{ useApi: true }, { useApi: false }].forEach(function(params) {
23
- describe(u.scenario('Compression tests', params), function() {
24
- const p = u.p(params, { compression: true });
25
-
26
- it('should backup animaldb to a compressed file', async function() {
27
- // Allow up to 60 s for backup of animaldb
28
- u.setTimeout(this, 60);
29
- const compressedBackup = `./${this.fileName}`;
30
- const output = fs.createWriteStream(compressedBackup);
31
- return once(output, 'open')
32
- .then(() => {
33
- return u.testBackup(p, 'animaldb', output);
34
- }).then(() => {
35
- return u.assertGzipFile(compressedBackup);
36
- });
37
- });
25
+ const url = require('url');
38
26
 
39
- it('should backup and restore animaldb via a compressed file', async function() {
40
- // Allow up to 60 s for backup and restore of animaldb
41
- u.setTimeout(this, 60);
42
- const compressedBackup = `./${this.fileName}`;
43
- return u.testBackupAndRestoreViaFile(p, 'animaldb', compressedBackup, this.dbName).then(() => {
44
- return u.assertGzipFile(compressedBackup);
45
- });
46
- });
27
+ const AWS = require('aws-sdk');
28
+ const couchbackup = require('@cloudant/couchbackup');
29
+ const debug = require('debug')('s3-backup');
30
+ const tmp = require('tmp');
31
+ const VError = require('verror').VError;
32
+
33
+ /*
34
+ Main function, run from base of file.
35
+ */
36
+ function main() {
37
+ const argv = require('yargs')
38
+ .usage('Usage: $0 [options]')
39
+ .example('$0 -s https://user:pass@host/db -b <bucket>', 'Backup db to bucket')
40
+ .options({
41
+ source: { alias: 's', nargs: 1, demandOption: true, describe: 'Source database URL' },
42
+ bucket: { alias: 'b', nargs: 1, demandOption: true, describe: 'Destination bucket' },
43
+ prefix: { alias: 'p', nargs: 1, describe: 'Prefix for backup object key', default: 'couchbackup' },
44
+ s3url: { nargs: 1, describe: 'S3 endpoint URL' },
45
+ awsprofile: { nargs: 1, describe: 'The profile section to use in the ~/.aws/credentials file', default: 'default' }
46
+ })
47
+ .help('h').alias('h', 'help')
48
+ .epilog('Copyright (C) IBM 2017')
49
+ .argv;
47
50
 
48
- it('should backup and restore animaldb via a compressed stream', async function() {
49
- // Allow up to 60 s for backup and restore of animaldb
50
- u.setTimeout(this, 60);
51
- return u.testDirectBackupAndRestore(p, 'animaldb', this.dbName);
51
+ const sourceUrl = argv.source;
52
+ const backupBucket = argv.bucket;
53
+ const backupName = new url.URL(sourceUrl).pathname.split('/').filter(function(x) { return x; }).join('-');
54
+ const backupKeyPrefix = `${argv.prefix}-${backupName}`;
55
+
56
+ const backupKey = `${backupKeyPrefix}-${new Date().toISOString()}`;
57
+ const backupTmpFile = tmp.fileSync();
58
+
59
+ const s3Endpoint = argv.s3url;
60
+ const awsProfile = argv.awsprofile;
61
+
62
+ // Creds are from ~/.aws/credentials, environment etc. (see S3 docs).
63
+ const awsOpts = {
64
+ signatureVersion: 'v4',
65
+ credentials: new AWS.SharedIniFileCredentials({ profile: awsProfile })
66
+ };
67
+ if (typeof s3Endpoint !== 'undefined') {
68
+ awsOpts.endpoint = new AWS.Endpoint(s3Endpoint);
69
+ }
70
+ const s3 = new AWS.S3(awsOpts);
71
+
72
+ debug(`Creating a new backup of ${s(sourceUrl)} at ${backupBucket}/${backupKey}...`);
73
+ bucketAccessible(s3, backupBucket)
74
+ .then(() => {
75
+ return createBackupFile(sourceUrl, backupTmpFile.name);
76
+ })
77
+ .then(() => {
78
+ return uploadNewBackup(s3, backupTmpFile.name, backupBucket, backupKey);
79
+ })
80
+ .then(() => {
81
+ debug('Backup successful!');
82
+ backupTmpFile.removeCallback();
83
+ debug('done.');
84
+ })
85
+ .catch((reason) => {
86
+ debug(`Error: ${reason}`);
52
87
  });
88
+ }
53
89
 
54
- it('should backup and restore largedb2g via a compressed file #slower', async function() {
55
- // Takes ~ 25 min using CLI, but sometimes over an hour with API
56
- u.setTimeout(this, 180 * 60);
57
- const compressedBackup = `./${this.fileName}`;
58
- params.compression = true;
59
- return u.testBackupAndRestoreViaFile(p, 'largedb2g', compressedBackup, this.dbName);
90
+ /**
91
+ * Return a promise that resolves if the bucket is available and
92
+ * rejects if not.
93
+ *
94
+ * @param {any} s3 S3 client object
95
+ * @param {any} bucketName Bucket name
96
+ * @returns Promise
97
+ */
98
+ function bucketAccessible(s3, bucketName) {
99
+ return new Promise(function(resolve, reject) {
100
+ const params = {
101
+ Bucket: bucketName
102
+ };
103
+ s3.headBucket(params, function(err, data) {
104
+ if (err) {
105
+ reject(new VError(err, 'S3 bucket not accessible'));
106
+ } else {
107
+ resolve();
108
+ }
60
109
  });
61
110
  });
62
- });
111
+ }
112
+
113
+ /**
114
+ * Use couchbackup to create a backup of the specified database to a file path.
115
+ *
116
+ * @param {any} sourceUrl Database URL
117
+ * @param {any} backupTmpFilePath Path to write file
118
+ * @returns Promise
119
+ */
120
+ function createBackupFile(sourceUrl, backupTmpFilePath) {
121
+ return new Promise((resolve, reject) => {
122
+ couchbackup.backup(
123
+ sourceUrl,
124
+ fs.createWriteStream(backupTmpFilePath),
125
+ (err) => {
126
+ if (err) {
127
+ return reject(new VError(err, 'CouchBackup process failed'));
128
+ }
129
+ debug('couchbackup to file done; uploading to S3');
130
+ resolve('creating backup file complete');
131
+ }
132
+ );
133
+ });
134
+ }
135
+
136
+ /**
137
+ * Upload a backup file to an S3 bucket.
138
+ *
139
+ * @param {any} s3 Object store client
140
+ * @param {any} backupTmpFilePath Path of backup file to write.
141
+ * @param {any} bucket Object store bucket name
142
+ * @param {any} key Object store key name
143
+ * @returns Promise
144
+ */
145
+ function uploadNewBackup(s3, backupTmpFilePath, bucket, key) {
146
+ return new Promise((resolve, reject) => {
147
+ debug(`Uploading from ${backupTmpFilePath} to ${bucket}/${key}`);
148
+
149
+ function uploadFromStream(s3, bucket, key) {
150
+ const pass = new stream.PassThrough();
151
+
152
+ const params = {
153
+ Bucket: bucket,
154
+ Key: key,
155
+ Body: pass
156
+ };
157
+ s3.upload(params, function(err, data) {
158
+ debug('S3 upload done');
159
+ if (err) {
160
+ debug(err);
161
+ reject(new VError(err, 'Upload failed'));
162
+ return;
163
+ }
164
+ debug('Upload succeeded');
165
+ debug(data);
166
+ resolve();
167
+ }).httpUploadProgress = (progress) => {
168
+ debug(`S3 upload progress: ${progress}`);
169
+ };
170
+
171
+ return pass;
172
+ }
173
+
174
+ const inputStream = fs.createReadStream(backupTmpFilePath);
175
+ const s3Stream = uploadFromStream(s3, bucket, key);
176
+ inputStream.pipe(s3Stream);
177
+ });
178
+ }
179
+
180
+ /**
181
+ * Remove creds from a URL, e.g., before logging
182
+ *
183
+ * @param {string} url URL to safen
184
+ */
185
+ function s(originalUrl) {
186
+ const parts = new url.URL(originalUrl);
187
+ return url.format(parts, { auth: false });
188
+ }
189
+
190
+ main();