test_sqlite.py 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. import logging
  2. from flexmock import flexmock
  3. from borgmatic.hooks.data_source import sqlite as module
  4. def test_use_streaming_true_for_any_databases():
  5. assert module.use_streaming(
  6. databases=[flexmock(), flexmock()],
  7. config=flexmock(),
  8. )
  9. def test_use_streaming_false_for_no_databases():
  10. assert not module.use_streaming(databases=[], config=flexmock())
  11. def test_dump_data_sources_logs_and_skips_if_dump_already_exists():
  12. databases = [{'path': '/path/to/database', 'name': 'database'}]
  13. flexmock(module).should_receive('make_dump_path').and_return('/run/borgmatic')
  14. flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return(
  15. '/run/borgmatic/database'
  16. )
  17. flexmock(module.os.path).should_receive('exists').and_return(True)
  18. flexmock(module.dump).should_receive('create_named_pipe_for_dump').never()
  19. flexmock(module).should_receive('execute_command').never()
  20. assert (
  21. module.dump_data_sources(
  22. databases,
  23. {},
  24. config_paths=('test.yaml',),
  25. borgmatic_runtime_directory='/run/borgmatic',
  26. patterns=[],
  27. dry_run=False,
  28. )
  29. == []
  30. )
  31. def test_dump_data_sources_dumps_each_database():
  32. databases = [
  33. {'path': '/path/to/database1', 'name': 'database1'},
  34. {'path': '/path/to/database2', 'name': 'database2'},
  35. ]
  36. processes = [flexmock(), flexmock()]
  37. flexmock(module).should_receive('make_dump_path').and_return('/run/borgmatic')
  38. flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return(
  39. '/run/borgmatic/database'
  40. )
  41. flexmock(module.os.path).should_receive('exists').and_return(False)
  42. flexmock(module.dump).should_receive('create_named_pipe_for_dump')
  43. flexmock(module).should_receive('execute_command').and_return(processes[0]).and_return(
  44. processes[1]
  45. )
  46. assert (
  47. module.dump_data_sources(
  48. databases,
  49. {},
  50. config_paths=('test.yaml',),
  51. borgmatic_runtime_directory='/run/borgmatic',
  52. patterns=[],
  53. dry_run=False,
  54. )
  55. == processes
  56. )
  57. def test_dump_data_sources_with_path_injection_attack_gets_escaped():
  58. databases = [
  59. {'path': '/path/to/database1; naughty-command', 'name': 'database1'},
  60. ]
  61. processes = [flexmock()]
  62. flexmock(module).should_receive('make_dump_path').and_return('/run/borgmatic')
  63. flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return(
  64. '/run/borgmatic/database'
  65. )
  66. flexmock(module.os.path).should_receive('exists').and_return(False)
  67. flexmock(module.dump).should_receive('create_named_pipe_for_dump')
  68. flexmock(module).should_receive('execute_command').with_args(
  69. (
  70. 'sqlite3',
  71. "'/path/to/database1; naughty-command'",
  72. '.dump',
  73. '>',
  74. '/run/borgmatic/database',
  75. ),
  76. shell=True,
  77. run_to_completion=False,
  78. ).and_return(processes[0])
  79. assert (
  80. module.dump_data_sources(
  81. databases,
  82. {},
  83. config_paths=('test.yaml',),
  84. borgmatic_runtime_directory='/run/borgmatic',
  85. patterns=[],
  86. dry_run=False,
  87. )
  88. == processes
  89. )
  90. def test_dump_data_sources_with_non_existent_path_warns_and_dumps_database():
  91. databases = [
  92. {'path': '/path/to/database1', 'name': 'database1'},
  93. ]
  94. processes = [flexmock()]
  95. flexmock(module).should_receive('make_dump_path').and_return('/run/borgmatic')
  96. flexmock(module.logger).should_receive('warning').once()
  97. flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return(
  98. '/run/borgmatic'
  99. )
  100. flexmock(module.os.path).should_receive('exists').and_return(False)
  101. flexmock(module.dump).should_receive('create_named_pipe_for_dump')
  102. flexmock(module).should_receive('execute_command').and_return(processes[0])
  103. assert (
  104. module.dump_data_sources(
  105. databases,
  106. {},
  107. config_paths=('test.yaml',),
  108. borgmatic_runtime_directory='/run/borgmatic',
  109. patterns=[],
  110. dry_run=False,
  111. )
  112. == processes
  113. )
  114. def test_dump_data_sources_with_name_all_warns_and_dumps_all_databases():
  115. databases = [
  116. {'path': '/path/to/database1', 'name': 'all'},
  117. ]
  118. processes = [flexmock()]
  119. flexmock(module).should_receive('make_dump_path').and_return('/run/borgmatic')
  120. flexmock(module.logger).should_receive(
  121. 'warning'
  122. ).twice() # once for the name=all, once for the non-existent path
  123. flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return(
  124. '/run/borgmatic/database'
  125. )
  126. flexmock(module.os.path).should_receive('exists').and_return(False)
  127. flexmock(module.dump).should_receive('create_named_pipe_for_dump')
  128. flexmock(module).should_receive('execute_command').and_return(processes[0])
  129. assert (
  130. module.dump_data_sources(
  131. databases,
  132. {},
  133. config_paths=('test.yaml',),
  134. borgmatic_runtime_directory='/run/borgmatic',
  135. patterns=[],
  136. dry_run=False,
  137. )
  138. == processes
  139. )
  140. def test_dump_data_sources_does_not_dump_if_dry_run():
  141. databases = [{'path': '/path/to/database', 'name': 'database'}]
  142. flexmock(module).should_receive('make_dump_path').and_return('/run/borgmatic')
  143. flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return(
  144. '/run/borgmatic'
  145. )
  146. flexmock(module.os.path).should_receive('exists').and_return(False)
  147. flexmock(module.dump).should_receive('create_named_pipe_for_dump').never()
  148. flexmock(module).should_receive('execute_command').never()
  149. assert (
  150. module.dump_data_sources(
  151. databases,
  152. {},
  153. config_paths=('test.yaml',),
  154. borgmatic_runtime_directory='/run/borgmatic',
  155. patterns=[],
  156. dry_run=True,
  157. )
  158. == []
  159. )
  160. def test_restore_data_source_dump_restores_database():
  161. hook_config = [{'path': '/path/to/database', 'name': 'database'}, {'name': 'other'}]
  162. extract_process = flexmock(stdout=flexmock())
  163. flexmock(module).should_receive('execute_command_with_processes').with_args(
  164. (
  165. 'sqlite3',
  166. '/path/to/database',
  167. ),
  168. processes=[extract_process],
  169. output_log_level=logging.DEBUG,
  170. input_file=extract_process.stdout,
  171. ).once()
  172. flexmock(module.os).should_receive('remove').once()
  173. module.restore_data_source_dump(
  174. hook_config,
  175. {},
  176. data_source=hook_config[0],
  177. dry_run=False,
  178. extract_process=extract_process,
  179. connection_params={'restore_path': None},
  180. borgmatic_runtime_directory='/run/borgmatic',
  181. )
  182. def test_restore_data_source_dump_with_connection_params_uses_connection_params_for_restore():
  183. hook_config = [
  184. {'path': '/path/to/database', 'name': 'database', 'restore_path': 'config/path/to/database'}
  185. ]
  186. extract_process = flexmock(stdout=flexmock())
  187. flexmock(module).should_receive('execute_command_with_processes').with_args(
  188. (
  189. 'sqlite3',
  190. 'cli/path/to/database',
  191. ),
  192. processes=[extract_process],
  193. output_log_level=logging.DEBUG,
  194. input_file=extract_process.stdout,
  195. ).once()
  196. flexmock(module.os).should_receive('remove').once()
  197. module.restore_data_source_dump(
  198. hook_config,
  199. {},
  200. data_source={'name': 'database'},
  201. dry_run=False,
  202. extract_process=extract_process,
  203. connection_params={'restore_path': 'cli/path/to/database'},
  204. borgmatic_runtime_directory='/run/borgmatic',
  205. )
  206. def test_restore_data_source_dump_without_connection_params_uses_restore_params_in_config_for_restore():
  207. hook_config = [
  208. {'path': '/path/to/database', 'name': 'database', 'restore_path': 'config/path/to/database'}
  209. ]
  210. extract_process = flexmock(stdout=flexmock())
  211. flexmock(module).should_receive('execute_command_with_processes').with_args(
  212. (
  213. 'sqlite3',
  214. 'config/path/to/database',
  215. ),
  216. processes=[extract_process],
  217. output_log_level=logging.DEBUG,
  218. input_file=extract_process.stdout,
  219. ).once()
  220. flexmock(module.os).should_receive('remove').once()
  221. module.restore_data_source_dump(
  222. hook_config,
  223. {},
  224. data_source=hook_config[0],
  225. dry_run=False,
  226. extract_process=extract_process,
  227. connection_params={'restore_path': None},
  228. borgmatic_runtime_directory='/run/borgmatic',
  229. )
  230. def test_restore_data_source_dump_does_not_restore_database_if_dry_run():
  231. hook_config = [{'path': '/path/to/database', 'name': 'database'}]
  232. extract_process = flexmock(stdout=flexmock())
  233. flexmock(module).should_receive('execute_command_with_processes').never()
  234. flexmock(module.os).should_receive('remove').never()
  235. module.restore_data_source_dump(
  236. hook_config,
  237. {},
  238. data_source={'name': 'database'},
  239. dry_run=True,
  240. extract_process=extract_process,
  241. connection_params={'restore_path': None},
  242. borgmatic_runtime_directory='/run/borgmatic',
  243. )