meerschaum 2.7.3__tar.gz → 2.7.5__tar.gz

Sign up to get free protection for your applications and to get access to all the features.
Files changed (276) hide show
  1. {meerschaum-2.7.3/meerschaum.egg-info → meerschaum-2.7.5}/PKG-INFO +1 -1
  2. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/sql.py +14 -4
  3. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/start.py +2 -2
  4. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_default.py +6 -0
  5. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_version.py +1 -1
  6. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_cli.py +18 -12
  7. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_create_engine.py +1 -0
  8. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_pipes.py +14 -4
  9. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_sql.py +100 -15
  10. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/jobs/__init__.py +0 -2
  11. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/daemon/Daemon.py +7 -3
  12. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/daemon/StdinFile.py +4 -1
  13. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/packages/__init__.py +4 -1
  14. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/venv/_Venv.py +6 -1
  15. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/venv/__init__.py +39 -22
  16. {meerschaum-2.7.3 → meerschaum-2.7.5/meerschaum.egg-info}/PKG-INFO +1 -1
  17. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_jobs.py +3 -6
  18. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_pipes_dtypes.py +1 -3
  19. {meerschaum-2.7.3 → meerschaum-2.7.5}/LICENSE +0 -0
  20. {meerschaum-2.7.3 → meerschaum-2.7.5}/NOTICE +0 -0
  21. {meerschaum-2.7.3 → meerschaum-2.7.5}/README.md +0 -0
  22. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/__init__.py +0 -0
  23. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/__main__.py +0 -0
  24. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/__init__.py +0 -0
  25. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/arguments/__init__.py +0 -0
  26. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/arguments/_parse_arguments.py +0 -0
  27. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/arguments/_parser.py +0 -0
  28. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/docs/__init__.py +0 -0
  29. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/docs/index.py +0 -0
  30. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/entry.py +0 -0
  31. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/gui/__init__.py +0 -0
  32. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/gui/app/__init__.py +0 -0
  33. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/gui/app/_windows.py +0 -0
  34. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/gui/app/actions.py +0 -0
  35. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/gui/app/pipes.py +0 -0
  36. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/shell/Shell.py +0 -0
  37. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/shell/ShellCompleter.py +0 -0
  38. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/shell/ValidAutoSuggest.py +0 -0
  39. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/shell/__init__.py +0 -0
  40. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/shell/resources/__init__.py +0 -0
  41. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/shell/updates.py +0 -0
  42. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/term/TermPageHandler.py +0 -0
  43. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/term/__init__.py +0 -0
  44. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/_internal/term/tools.py +0 -0
  45. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/__init__.py +0 -0
  46. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/api.py +0 -0
  47. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/attach.py +0 -0
  48. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/bootstrap.py +0 -0
  49. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/clear.py +0 -0
  50. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/copy.py +0 -0
  51. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/deduplicate.py +0 -0
  52. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/delete.py +0 -0
  53. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/drop.py +0 -0
  54. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/edit.py +0 -0
  55. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/install.py +0 -0
  56. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/login.py +0 -0
  57. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/os.py +0 -0
  58. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/pause.py +0 -0
  59. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/python.py +0 -0
  60. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/register.py +0 -0
  61. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/reload.py +0 -0
  62. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/restart.py +0 -0
  63. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/setup.py +0 -0
  64. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/sh.py +0 -0
  65. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/show.py +0 -0
  66. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/stack.py +0 -0
  67. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/stop.py +0 -0
  68. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/sync.py +0 -0
  69. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/tag.py +0 -0
  70. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/uninstall.py +0 -0
  71. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/upgrade.py +0 -0
  72. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/actions/verify.py +0 -0
  73. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/__init__.py +0 -0
  74. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/_chain.py +0 -0
  75. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/_events.py +0 -0
  76. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/_oauth2.py +0 -0
  77. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/_websockets.py +0 -0
  78. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/__init__.py +0 -0
  79. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/assets/__init__.py +0 -0
  80. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/assets/ansi_up.js +0 -0
  81. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/assets/banner_1920x320.png +0 -0
  82. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/assets/favicon.ico +0 -0
  83. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/assets/logo_48x48.png +0 -0
  84. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/assets/logo_500x500.png +0 -0
  85. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/callbacks/__init__.py +0 -0
  86. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/callbacks/custom.py +0 -0
  87. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/callbacks/dashboard.py +0 -0
  88. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/callbacks/jobs.py +0 -0
  89. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/callbacks/login.py +0 -0
  90. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/callbacks/pipes.py +0 -0
  91. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/callbacks/plugins.py +0 -0
  92. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/callbacks/register.py +0 -0
  93. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/components.py +0 -0
  94. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/connectors.py +0 -0
  95. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/graphs.py +0 -0
  96. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/jobs.py +0 -0
  97. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/keys.py +0 -0
  98. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pages/__init__.py +0 -0
  99. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pages/dashboard.py +0 -0
  100. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pages/error.py +0 -0
  101. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pages/job.py +0 -0
  102. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pages/login.py +0 -0
  103. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pages/pipes.py +0 -0
  104. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pages/plugins.py +0 -0
  105. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pages/register.py +0 -0
  106. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/pipes.py +0 -0
  107. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/plugins.py +0 -0
  108. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/sessions.py +0 -0
  109. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/sync.py +0 -0
  110. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/users.py +0 -0
  111. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/websockets.py +0 -0
  112. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/dash/webterm.py +0 -0
  113. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/models/__init__.py +0 -0
  114. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/models/_interfaces.py +0 -0
  115. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/models/_locations.py +0 -0
  116. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/models/_metrics.py +0 -0
  117. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/models/_pipes.py +0 -0
  118. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/__init__.py +0 -0
  119. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/__init__.py +0 -0
  120. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/css/__init__.py +0 -0
  121. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/css/bootstrap.min.css +0 -0
  122. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/css/dash.css +0 -0
  123. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/css/dbc_dark.css +0 -0
  124. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/css/styles.css +0 -0
  125. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/css/xterm.css +0 -0
  126. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/ico/__init__.py +0 -0
  127. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/ico/logo.ico +0 -0
  128. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/js/__init__.py +0 -0
  129. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/js/action_button.js +0 -0
  130. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/js/main.js +0 -0
  131. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/js/terminado.js +0 -0
  132. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/js/xterm.js +0 -0
  133. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/static/png/__init__.py +0 -0
  134. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/templates/__init__.py +0 -0
  135. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/templates/index.html +0 -0
  136. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/templates/old_index.html +0 -0
  137. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/templates/secret.html +0 -0
  138. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/resources/templates/termpage.html +0 -0
  139. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/__init__.py +0 -0
  140. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_actions.py +0 -0
  141. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_connectors.py +0 -0
  142. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_index.py +0 -0
  143. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_jobs.py +0 -0
  144. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_login.py +0 -0
  145. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_misc.py +0 -0
  146. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_pipes.py +0 -0
  147. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_plugins.py +0 -0
  148. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_users.py +0 -0
  149. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_version.py +0 -0
  150. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/routes/_webterm.py +0 -0
  151. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/api/tables/__init__.py +0 -0
  152. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/__init__.py +0 -0
  153. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_dash.py +0 -0
  154. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_edit.py +0 -0
  155. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_environment.py +0 -0
  156. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_formatting.py +0 -0
  157. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_jobs.py +0 -0
  158. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_patch.py +0 -0
  159. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_paths.py +0 -0
  160. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_preprocess.py +0 -0
  161. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_read_config.py +0 -0
  162. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_shell.py +0 -0
  163. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/_sync.py +0 -0
  164. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/paths.py +0 -0
  165. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/resources/__init__.py +0 -0
  166. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/stack/__init__.py +0 -0
  167. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/stack/grafana/__init__.py +0 -0
  168. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/stack/mosquitto/__init__.py +0 -0
  169. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/stack/mosquitto/resources/__init__.py +0 -0
  170. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/stack/resources/__init__.py +0 -0
  171. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/config/static/__init__.py +0 -0
  172. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/_Connector.py +0 -0
  173. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/__init__.py +0 -0
  174. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_APIConnector.py +0 -0
  175. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/__init__.py +0 -0
  176. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_actions.py +0 -0
  177. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_fetch.py +0 -0
  178. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_jobs.py +0 -0
  179. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_login.py +0 -0
  180. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_misc.py +0 -0
  181. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_pipes.py +0 -0
  182. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_plugins.py +0 -0
  183. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_request.py +0 -0
  184. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_uri.py +0 -0
  185. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/api/_users.py +0 -0
  186. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/parse.py +0 -0
  187. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/plugin/PluginConnector.py +0 -0
  188. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/plugin/__init__.py +0 -0
  189. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/poll.py +0 -0
  190. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_SQLConnector.py +0 -0
  191. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/__init__.py +0 -0
  192. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_fetch.py +0 -0
  193. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_instance.py +0 -0
  194. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_plugins.py +0 -0
  195. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_uri.py +0 -0
  196. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/_users.py +0 -0
  197. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/tables/__init__.py +0 -0
  198. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/tables/types.py +0 -0
  199. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/sql/tools.py +0 -0
  200. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/valkey/_ValkeyConnector.py +0 -0
  201. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/valkey/__init__.py +0 -0
  202. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/valkey/_fetch.py +0 -0
  203. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/valkey/_pipes.py +0 -0
  204. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/valkey/_plugins.py +0 -0
  205. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/connectors/valkey/_users.py +0 -0
  206. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/__init__.py +0 -0
  207. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_attributes.py +0 -0
  208. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_bootstrap.py +0 -0
  209. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_clear.py +0 -0
  210. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_copy.py +0 -0
  211. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_data.py +0 -0
  212. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_deduplicate.py +0 -0
  213. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_delete.py +0 -0
  214. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_drop.py +0 -0
  215. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_dtypes.py +0 -0
  216. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_edit.py +0 -0
  217. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_fetch.py +0 -0
  218. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_register.py +0 -0
  219. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_show.py +0 -0
  220. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_sync.py +0 -0
  221. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Pipe/_verify.py +0 -0
  222. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/Plugin/__init__.py +0 -0
  223. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/User/_User.py +0 -0
  224. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/User/__init__.py +0 -0
  225. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/core/__init__.py +0 -0
  226. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/jobs/_Executor.py +0 -0
  227. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/jobs/_Job.py +0 -0
  228. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/jobs/systemd.py +0 -0
  229. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/plugins/_Plugin.py +0 -0
  230. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/plugins/__init__.py +0 -0
  231. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/plugins/bootstrap.py +0 -0
  232. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/__init__.py +0 -0
  233. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/_get_pipes.py +0 -0
  234. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/daemon/FileDescriptorInterceptor.py +0 -0
  235. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/daemon/RotatingFile.py +0 -0
  236. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/daemon/__init__.py +0 -0
  237. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/daemon/_names.py +0 -0
  238. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/dataframe.py +0 -0
  239. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/debug.py +0 -0
  240. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/dtypes/__init__.py +0 -0
  241. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/dtypes/sql.py +0 -0
  242. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/formatting/__init__.py +0 -0
  243. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/formatting/_jobs.py +0 -0
  244. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/formatting/_pipes.py +0 -0
  245. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/formatting/_pprint.py +0 -0
  246. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/formatting/_shell.py +0 -0
  247. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/interactive.py +0 -0
  248. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/misc.py +0 -0
  249. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/networking.py +0 -0
  250. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/packages/_packages.py +0 -0
  251. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/packages/lazy_loader.py +0 -0
  252. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/pool.py +0 -0
  253. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/process.py +0 -0
  254. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/prompt.py +0 -0
  255. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/schedule.py +0 -0
  256. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/sql.py +0 -0
  257. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/threading.py +0 -0
  258. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/typing.py +0 -0
  259. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/warnings.py +0 -0
  260. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum/utils/yaml.py +0 -0
  261. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum.egg-info/SOURCES.txt +0 -0
  262. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum.egg-info/dependency_links.txt +0 -0
  263. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum.egg-info/entry_points.txt +0 -0
  264. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum.egg-info/requires.txt +0 -0
  265. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum.egg-info/top_level.txt +0 -0
  266. {meerschaum-2.7.3 → meerschaum-2.7.5}/meerschaum.egg-info/zip-safe +0 -0
  267. {meerschaum-2.7.3 → meerschaum-2.7.5}/setup.cfg +0 -0
  268. {meerschaum-2.7.3 → meerschaum-2.7.5}/setup.py +0 -0
  269. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_actions.py +0 -0
  270. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_arguments.py +0 -0
  271. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_deduplicate.py +0 -0
  272. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_pipe_data.py +0 -0
  273. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_sql.py +0 -0
  274. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_sync.py +0 -0
  275. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_users.py +0 -0
  276. {meerschaum-2.7.3 → meerschaum-2.7.5}/tests/test_verify.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: meerschaum
3
- Version: 2.7.3
3
+ Version: 2.7.5
4
4
  Summary: Sync Time-Series Pipes with Meerschaum
5
5
  Home-page: https://meerschaum.io
6
6
  Author: Bennett Meares
@@ -21,7 +21,7 @@ def sql(
21
21
  nopretty: bool = False,
22
22
  debug: bool = False,
23
23
  **kw: Any
24
- ):
24
+ ) -> SuccessTuple:
25
25
  """Execute a SQL query or launch an interactive CLI. All positional arguments are optional.
26
26
 
27
27
  Usage:
@@ -125,10 +125,9 @@ def sql(
125
125
  if result is False:
126
126
  return (False, f"Failed to execute query:\n\n{query}")
127
127
 
128
- from meerschaum.utils.packages import attempt_import, import_pandas
128
+ from meerschaum.utils.packages import attempt_import
129
129
  from meerschaum.utils.formatting import print_tuple, pprint
130
- sqlalchemy_engine_result = attempt_import('sqlalchemy.engine.result')
131
- pd = import_pandas()
130
+ _ = attempt_import('sqlalchemy.engine.result')
132
131
  if 'sqlalchemy' in str(type(result)):
133
132
  if not nopretty:
134
133
  print_tuple((True, f"Successfully executed query:\n\n{query}"))
@@ -145,3 +144,14 @@ def sql(
145
144
  )
146
145
 
147
146
  return (True, "Success")
147
+
148
+
149
+ def _complete_sql(
150
+ action: Optional[List[str]] = None, **kw: Any
151
+ ) -> List[str]:
152
+ from meerschaum.utils.misc import get_connector_labels
153
+ _text = action[0] if action else ""
154
+ return [
155
+ label.split('sql:', maxsplit=1)[-1]
156
+ for label in get_connector_labels('sql', search_term=_text, ignore_exact_match=True)
157
+ ]
@@ -183,7 +183,7 @@ def _start_jobs(
183
183
  ### Cannot find dameon_id
184
184
  else:
185
185
  msg = (
186
- f"Unknown job" + ('s' if len(action) != 1 else '') + ' '
186
+ "Unknown job" + ('s' if len(action) != 1 else '') + ' '
187
187
  + items_str(action, and_str='or') + '.'
188
188
  )
189
189
  return False, msg
@@ -216,7 +216,7 @@ def _start_jobs(
216
216
  return job.start(debug=debug), name
217
217
 
218
218
  def _run_existing_job(name: str):
219
- job = Job(name, executor_keys=executor_keys)
219
+ job = jobs.get(name, Job(name, executor_keys=executor_keys))
220
220
  return job.start(debug=debug), name
221
221
 
222
222
  if not names:
@@ -68,6 +68,12 @@ default_system_config = {
68
68
  'pandas': 'pandas',
69
69
  },
70
70
  'sql': {
71
+ 'bulk_insert': {
72
+ 'postgresql': True,
73
+ 'citus': True,
74
+ 'timescaledb': True,
75
+ 'mssql': True,
76
+ },
71
77
  'instance': {
72
78
  'stale_temporary_tables_minutes': 1440,
73
79
  },
@@ -2,4 +2,4 @@
2
2
  Specify the Meerschaum release version.
3
3
  """
4
4
 
5
- __version__ = "2.7.3"
5
+ __version__ = "2.7.5"
@@ -26,46 +26,52 @@ flavor_clis = {
26
26
  'duckdb' : 'gadwall',
27
27
  }
28
28
  cli_deps = {
29
- 'pgcli': ['pgspecial', 'pendulum'],
29
+ 'pgcli': ['pgspecial', 'pendulum', 'cli_helpers'],
30
30
  'mycli': ['cryptography'],
31
+ 'mssql': ['cli_helpers'],
31
32
  }
32
33
 
33
34
 
34
35
  def cli(
35
- self,
36
- debug: bool = False,
37
- ) -> SuccessTuple:
36
+ self,
37
+ debug: bool = False,
38
+ ) -> SuccessTuple:
38
39
  """
39
40
  Launch a subprocess for an interactive CLI.
40
41
  """
42
+ from meerschaum.utils.warnings import dprint
41
43
  from meerschaum.utils.venv import venv_exec
42
44
  env = copy.deepcopy(dict(os.environ))
43
- env[f'MRSM_SQL_{self.label.upper()}'] = json.dumps(self.meta)
45
+ env_key = f"MRSM_SQL_{self.label.upper()}"
46
+ env_val = json.dumps(self.meta)
47
+ env[env_key] = env_val
44
48
  cli_code = (
45
49
  "import sys\n"
46
50
  "import meerschaum as mrsm\n"
51
+ "import os\n"
47
52
  f"conn = mrsm.get_connector('sql:{self.label}')\n"
48
53
  "success, msg = conn._cli_exit()\n"
49
54
  "mrsm.pprint((success, msg))\n"
50
55
  "if not success:\n"
51
56
  " raise Exception(msg)"
52
57
  )
58
+ if debug:
59
+ dprint(cli_code)
53
60
  try:
54
- _ = venv_exec(cli_code, venv=None, debug=debug, capture_output=False)
61
+ _ = venv_exec(cli_code, venv=None, env=env, debug=debug, capture_output=False)
55
62
  except Exception as e:
56
63
  return False, f"[{self}] Failed to start CLI:\n{e}"
57
64
  return True, "Success"
58
65
 
59
66
 
60
67
  def _cli_exit(
61
- self,
62
- debug: bool = False
63
- ) -> SuccessTuple:
68
+ self,
69
+ debug: bool = False
70
+ ) -> SuccessTuple:
64
71
  """Launch an interactive CLI for the SQLConnector's flavor."""
65
- from meerschaum.utils.packages import venv_exec, attempt_import
72
+ import os
73
+ from meerschaum.utils.packages import attempt_import
66
74
  from meerschaum.utils.debug import dprint
67
- from meerschaum.utils.warnings import error
68
- import sys, subprocess, os
69
75
 
70
76
  if self.flavor not in flavor_clis:
71
77
  return False, f"No CLI available for flavor '{self.flavor}'."
@@ -61,6 +61,7 @@ flavor_configs = {
61
61
  'engine': 'mssql+pyodbc',
62
62
  'create_engine': {
63
63
  'fast_executemany': True,
64
+ 'use_insertmanyvalues': False,
64
65
  'isolation_level': 'AUTOCOMMIT',
65
66
  'use_setinputsizes': False,
66
67
  'pool_pre_ping': True,
@@ -3064,7 +3064,12 @@ def get_alter_columns_queries(
3064
3064
  return []
3065
3065
  if pipe.static:
3066
3066
  return
3067
- from meerschaum.utils.sql import sql_item_name, DROP_IF_EXISTS_FLAVORS, get_table_cols_types
3067
+ from meerschaum.utils.sql import (
3068
+ sql_item_name,
3069
+ get_table_cols_types,
3070
+ DROP_IF_EXISTS_FLAVORS,
3071
+ SINGLE_ALTER_TABLE_FLAVORS,
3072
+ )
3068
3073
  from meerschaum.utils.dataframe import get_numeric_cols
3069
3074
  from meerschaum.utils.dtypes import are_dtypes_equal
3070
3075
  from meerschaum.utils.dtypes.sql import (
@@ -3308,14 +3313,19 @@ def get_alter_columns_queries(
3308
3313
  else 'TYPE '
3309
3314
  )
3310
3315
  column_str = 'COLUMN' if self.flavor != 'oracle' else ''
3311
- query += (
3316
+ query_suffix = (
3312
3317
  f"\n{alter_col_prefix} {column_str} "
3313
3318
  + sql_item_name(col, self.flavor, None)
3314
3319
  + " " + type_prefix + typ + ","
3315
3320
  )
3321
+ if self.flavor not in SINGLE_ALTER_TABLE_FLAVORS:
3322
+ query += query_suffix
3323
+ else:
3324
+ queries.append(query + query_suffix[:-1])
3325
+
3326
+ if self.flavor not in SINGLE_ALTER_TABLE_FLAVORS:
3327
+ queries.append(query[:-1])
3316
3328
 
3317
- query = query[:-1]
3318
- queries.append(query)
3319
3329
  if self.flavor != 'duckdb':
3320
3330
  return queries
3321
3331
 
@@ -6,8 +6,10 @@ This module contains SQLConnector functions for executing SQL queries.
6
6
  """
7
7
 
8
8
  from __future__ import annotations
9
+
10
+ import meerschaum as mrsm
9
11
  from meerschaum.utils.typing import (
10
- Union, Mapping, List, Dict, SuccessTuple, Optional, Any, Iterable, Callable,
12
+ Union, List, Dict, SuccessTuple, Optional, Any, Iterable, Callable,
11
13
  Tuple, Hashable,
12
14
  )
13
15
 
@@ -15,7 +17,7 @@ from meerschaum.utils.debug import dprint
15
17
  from meerschaum.utils.warnings import warn
16
18
 
17
19
  ### database flavors that can use bulk insert
18
- _bulk_flavors = {'postgresql', 'timescaledb', 'citus'}
20
+ _bulk_flavors = {'postgresql', 'timescaledb', 'citus', 'mssql'}
19
21
  ### flavors that do not support chunks
20
22
  _disallow_chunks_flavors = ['duckdb']
21
23
  _max_chunks_flavors = {'sqlite': 1000}
@@ -779,6 +781,7 @@ def to_sql(
779
781
  from meerschaum.utils.warnings import error, warn
780
782
  import warnings
781
783
  import functools
784
+
782
785
  if name is None:
783
786
  error(f"Name must not be `None` to insert data into {self}.")
784
787
 
@@ -805,6 +808,7 @@ def to_sql(
805
808
  quantize_decimal,
806
809
  coerce_timezone,
807
810
  encode_bytes_for_bytea,
811
+ serialize_bytes,
808
812
  )
809
813
  from meerschaum.utils.dtypes.sql import (
810
814
  NUMERIC_PRECISION_FLAVORS,
@@ -821,24 +825,36 @@ def to_sql(
821
825
  bytes_cols = get_bytes_cols(df)
822
826
  numeric_cols = get_numeric_cols(df)
823
827
 
824
- stats = {'target': name,}
828
+ enable_bulk_insert = mrsm.get_config(
829
+ 'system', 'connectors', 'sql', 'bulk_insert'
830
+ ).get(self.flavor, False)
831
+ stats = {'target': name}
825
832
  ### resort to defaults if None
826
833
  copied = False
827
- use_psql_copy = False
834
+ use_bulk_insert = False
828
835
  if method == "":
829
- if self.flavor in _bulk_flavors:
830
- method = functools.partial(psql_insert_copy, schema=self.schema)
831
- use_psql_copy = True
836
+ if enable_bulk_insert:
837
+ method = (
838
+ functools.partial(mssql_insert_json, debug=debug)
839
+ if self.flavor == 'mssql'
840
+ else functools.partial(psql_insert_copy, debug=debug)
841
+ )
842
+ use_bulk_insert = True
832
843
  else:
833
844
  ### Should resolve to 'multi' or `None`.
834
845
  method = flavor_configs.get(self.flavor, {}).get('to_sql', {}).get('method', 'multi')
835
846
 
836
- if bytes_cols and (use_psql_copy or self.flavor == 'oracle'):
847
+ if bytes_cols and (use_bulk_insert or self.flavor == 'oracle'):
837
848
  if safe_copy and not copied:
838
849
  df = df.copy()
839
850
  copied = True
851
+ bytes_serializer = (
852
+ functools.partial(encode_bytes_for_bytea, with_prefix=(self.flavor != 'oracle'))
853
+ if self.flavor != 'mssql'
854
+ else serialize_bytes
855
+ )
840
856
  for col in bytes_cols:
841
- df[col] = df[col].apply(encode_bytes_for_bytea, with_prefix=(self.flavor != 'oracle'))
857
+ df[col] = df[col].apply(bytes_serializer)
842
858
 
843
859
  if self.flavor in NUMERIC_AS_TEXT_FLAVORS:
844
860
  if safe_copy and not copied:
@@ -988,7 +1004,7 @@ def to_sql(
988
1004
  stats['duration'] = end - start
989
1005
 
990
1006
  if debug:
991
- print(f" done.", flush=True)
1007
+ print(" done.", flush=True)
992
1008
  dprint(msg)
993
1009
 
994
1010
  stats['success'] = success
@@ -1005,7 +1021,7 @@ def psql_insert_copy(
1005
1021
  conn: Union[sqlalchemy.engine.Engine, sqlalchemy.engine.Connection],
1006
1022
  keys: List[str],
1007
1023
  data_iter: Iterable[Any],
1008
- schema: Optional[str] = None,
1024
+ debug: bool = False,
1009
1025
  ) -> None:
1010
1026
  """
1011
1027
  Execute SQL statement inserting data for PostgreSQL.
@@ -1022,18 +1038,15 @@ def psql_insert_copy(
1022
1038
  data_iter: Iterable[Any]
1023
1039
  Iterable that iterates the values to be inserted
1024
1040
 
1025
- schema: Optional[str], default None
1026
- Optionally specify the schema of the table to be inserted into.
1027
-
1028
1041
  Returns
1029
1042
  -------
1030
1043
  None
1031
1044
  """
1032
1045
  import csv
1033
- from io import StringIO
1034
1046
  import json
1035
1047
 
1036
1048
  from meerschaum.utils.sql import sql_item_name
1049
+ from meerschaum.utils.warnings import dprint
1037
1050
 
1038
1051
  ### NOTE: PostgreSQL doesn't support NUL chars in text, so they're removed from strings.
1039
1052
  data_iter = (
@@ -1057,6 +1070,8 @@ def psql_insert_copy(
1057
1070
  table_name = sql_item_name(table.name, 'postgresql', table.schema)
1058
1071
  columns = ', '.join(f'"{k}"' for k in keys)
1059
1072
  sql = f"COPY {table_name} ({columns}) FROM STDIN WITH CSV NULL '\\N'"
1073
+ if debug:
1074
+ dprint(sql)
1060
1075
 
1061
1076
  dbapi_conn = conn.connection
1062
1077
  with dbapi_conn.cursor() as cur:
@@ -1065,6 +1080,76 @@ def psql_insert_copy(
1065
1080
  writer.writerows(data_iter)
1066
1081
 
1067
1082
 
1083
+ def mssql_insert_json(
1084
+ table: pandas.io.sql.SQLTable,
1085
+ conn: Union[sqlalchemy.engine.Engine, sqlalchemy.engine.Connection],
1086
+ keys: List[str],
1087
+ data_iter: Iterable[Any],
1088
+ cols_types: Optional[Dict[str, str]] = None,
1089
+ debug: bool = False,
1090
+ ):
1091
+ """
1092
+ Execute SQL statement inserting data via OPENJSON.
1093
+
1094
+ Adapted from this snippet:
1095
+ https://gist.github.com/gordthompson/1fb0f1c3f5edbf6192e596de8350f205
1096
+
1097
+ Parameters
1098
+ ----------
1099
+ table: pandas.io.sql.SQLTable
1100
+
1101
+ conn: Union[sqlalchemy.engine.Engine, sqlalchemy.engine.Connection]
1102
+
1103
+ keys: List[str]
1104
+ Column names
1105
+
1106
+ data_iter: Iterable[Any]
1107
+ Iterable that iterates the values to be inserted
1108
+
1109
+ cols_types: Optional[Dict[str, str]], default None
1110
+ If provided, use these as the columns and types for the table.
1111
+
1112
+ Returns
1113
+ -------
1114
+ None
1115
+ """
1116
+ import json
1117
+ from meerschaum.utils.sql import sql_item_name
1118
+ from meerschaum.utils.dtypes.sql import get_pd_type_from_db_type, get_db_type_from_pd_type
1119
+ from meerschaum.utils.warnings import dprint
1120
+ table_name = sql_item_name(table.name, 'mssql', table.schema)
1121
+ if not cols_types:
1122
+ pd_types = {
1123
+ str(column.name): get_pd_type_from_db_type(str(column.type))
1124
+ for column in table.table.columns
1125
+ }
1126
+ cols_types = {
1127
+ col: get_db_type_from_pd_type(typ, 'mssql')
1128
+ for col, typ in pd_types.items()
1129
+ }
1130
+ columns = ",\n ".join([f"[{k}]" for k in keys])
1131
+ json_data = [dict(zip(keys, row)) for row in data_iter]
1132
+ with_clause = ",\n ".join(
1133
+ [
1134
+ f"[{col_name}] {col_type} '$.\"{col_name}\"'"
1135
+ for col_name, col_type in cols_types.items()
1136
+ ]
1137
+ )
1138
+ placeholder = "?" if conn.dialect.paramstyle == "qmark" else "%s"
1139
+ sql = (
1140
+ f"INSERT INTO {table_name} (\n {columns}\n)\n"
1141
+ f"SELECT\n {columns}\n"
1142
+ f"FROM OPENJSON({placeholder})\n"
1143
+ "WITH (\n"
1144
+ f" {with_clause}\n"
1145
+ ");"
1146
+ )
1147
+ if debug:
1148
+ dprint(sql)
1149
+
1150
+ conn.exec_driver_sql(sql, (json.dumps(json_data, default=str),))
1151
+
1152
+
1068
1153
  def format_sql_query_for_dask(query: str) -> 'sqlalchemy.sql.selectable.Select':
1069
1154
  """
1070
1155
  Given a `SELECT` query, return a `sqlalchemy` query for Dask to use.
@@ -6,8 +6,6 @@
6
6
  Higher-level utilities for managing `meerschaum.utils.daemon.Daemon`.
7
7
  """
8
8
 
9
- import pathlib
10
-
11
9
  import meerschaum as mrsm
12
10
  from meerschaum.utils.typing import Dict, Optional, List, SuccessTuple
13
11
 
@@ -13,7 +13,6 @@ import pathlib
13
13
  import json
14
14
  import shutil
15
15
  import signal
16
- import sys
17
16
  import time
18
17
  import traceback
19
18
  from functools import partial
@@ -262,7 +261,10 @@ class Daemon:
262
261
  -------
263
262
  Nothing — this will exit the parent process.
264
263
  """
265
- import platform, sys, os, traceback
264
+ import platform
265
+ import sys
266
+ import os
267
+ import traceback
266
268
  from meerschaum.utils.warnings import warn
267
269
  from meerschaum.config import get_config
268
270
  daemon = attempt_import('daemon')
@@ -301,6 +303,7 @@ class Daemon:
301
303
  os.environ['LINES'], os.environ['COLUMNS'] = str(int(lines)), str(int(columns))
302
304
  with self._daemon_context:
303
305
  sys.stdin = self.stdin_file
306
+ _ = os.environ.pop(STATIC_CONFIG['environment']['systemd_stdin_path'], None)
304
307
  os.environ[STATIC_CONFIG['environment']['daemon_id']] = self.daemon_id
305
308
  os.environ['PYTHONUNBUFFERED'] = '1'
306
309
 
@@ -1085,7 +1088,8 @@ class Daemon:
1085
1088
 
1086
1089
  def write_pickle(self) -> SuccessTuple:
1087
1090
  """Write the pickle file for the daemon."""
1088
- import pickle, traceback
1091
+ import pickle
1092
+ import traceback
1089
1093
  try:
1090
1094
  self.path.mkdir(parents=True, exist_ok=True)
1091
1095
  with open(self.pickle_path, 'wb+') as pickle_file:
@@ -104,7 +104,10 @@ class StdinFile(io.TextIOBase):
104
104
  if self._file_handler is not None:
105
105
  self.sel.unregister(self._file_handler)
106
106
  self._file_handler.close()
107
- os.close(self._fd)
107
+ try:
108
+ os.close(self._fd)
109
+ except OSError:
110
+ pass
108
111
  self._file_handler = None
109
112
  self._fd = None
110
113
 
@@ -831,7 +831,6 @@ def pip_install(
831
831
 
832
832
  """
833
833
  from meerschaum.config._paths import VIRTENV_RESOURCES_PATH
834
- from meerschaum.config import get_config
835
834
  from meerschaum.config.static import STATIC_CONFIG
836
835
  from meerschaum.utils.warnings import warn
837
836
  if args is None:
@@ -966,6 +965,10 @@ def pip_install(
966
965
 
967
966
  if '--target' not in _args and '-t' not in _args and not (not use_uv_pip and _uninstall):
968
967
  if venv is not None:
968
+ vtp = venv_target_path(venv, allow_nonexistent=True, debug=debug)
969
+ if not vtp.exists():
970
+ if not init_venv(venv, force=True):
971
+ vtp.mkdir(parents=True, exist_ok=True)
969
972
  _args += ['--target', venv_target_path(venv, debug=debug)]
970
973
  elif (
971
974
  '--target' not in _args
@@ -62,8 +62,13 @@ class Venv:
62
62
  If a `meerschaum.plugins.Plugin` was provided, its dependent virtual environments
63
63
  will also be activated.
64
64
  """
65
- from meerschaum.utils.venv import active_venvs
65
+ from meerschaum.utils.venv import active_venvs, init_venv
66
66
  self._kwargs['previously_active_venvs'] = copy.deepcopy(active_venvs)
67
+ try:
68
+ return self._activate(debug=(debug or self._debug), **self._kwargs)
69
+ except OSError as e:
70
+ if not init_venv(self._venv, force=True):
71
+ raise e
67
72
  return self._activate(debug=(debug or self._debug), **self._kwargs)
68
73
 
69
74
 
@@ -176,23 +176,26 @@ def deactivate_venv(
176
176
  if sys.path is None:
177
177
  return False
178
178
 
179
- target = venv_target_path(venv, allow_nonexistent=force, debug=debug).as_posix()
179
+ target = venv_target_path(venv, allow_nonexistent=True, debug=debug).as_posix()
180
180
  with LOCKS['sys.path']:
181
181
  if target in sys.path:
182
- sys.path.remove(target)
182
+ try:
183
+ sys.path.remove(target)
184
+ except Exception:
185
+ pass
183
186
  try:
184
187
  active_venvs_order.remove(venv)
185
- except Exception as e:
188
+ except Exception:
186
189
  pass
187
190
 
188
191
  return True
189
192
 
190
193
 
191
194
  def is_venv_active(
192
- venv: str = 'mrsm',
193
- color : bool = True,
194
- debug: bool = False
195
- ) -> bool:
195
+ venv: str = 'mrsm',
196
+ color : bool = True,
197
+ debug: bool = False
198
+ ) -> bool:
196
199
  """
197
200
  Check if a virtual environment is active.
198
201
 
@@ -395,9 +398,13 @@ def init_venv(
395
398
 
396
399
  def update_lock(active: bool):
397
400
  try:
398
- if active:
401
+ if not active:
402
+ if debug:
403
+ print(f"Releasing lock: '{lock_path}'")
399
404
  lock_path.unlink()
400
405
  else:
406
+ if debug:
407
+ print(f"Acquiring lock: '{lock_path}'")
401
408
  lock_path.touch()
402
409
  except Exception:
403
410
  pass
@@ -406,9 +413,12 @@ def init_venv(
406
413
  max_lock_seconds = 1.0
407
414
  step_sleep_seconds = 0.1
408
415
  init_venv_check_start = time.perf_counter()
409
- while (time.perf_counter() - init_venv_check_start < max_lock_seconds):
416
+ while ((time.perf_counter() - init_venv_check_start) < max_lock_seconds):
410
417
  if not lock_path.exists():
411
- continue
418
+ break
419
+
420
+ if debug:
421
+ print(f"Lock exists for '{venv}', sleeping...")
412
422
  time.sleep(step_sleep_seconds)
413
423
  update_lock(False)
414
424
 
@@ -441,10 +451,9 @@ def init_venv(
441
451
  rename_vtp = vtp.exists() and not temp_vtp.exists()
442
452
 
443
453
  if rename_vtp:
444
- try:
445
- vtp.rename(temp_vtp)
446
- except FileExistsError:
447
- pass
454
+ if debug:
455
+ print(f"Moving '{vtp}' to '{temp_vtp}'...")
456
+ shutil.move(vtp, temp_vtp)
448
457
 
449
458
  wait_for_lock()
450
459
  update_lock(True)
@@ -491,7 +500,9 @@ def init_venv(
491
500
  + "Please install `virtualenv` via pip then restart Meerschaum."
492
501
  )
493
502
  if rename_vtp and temp_vtp.exists():
494
- temp_vtp.rename(vtp)
503
+ if debug:
504
+ print(f"Moving '{temp_vtp}' back to '{vtp}'...")
505
+ shutil.move(temp_vtp, vtp)
495
506
  update_lock(False)
496
507
  return False
497
508
 
@@ -530,7 +541,7 @@ def init_venv(
530
541
  import traceback
531
542
  traceback.print_exc()
532
543
  if rename_vtp and temp_vtp.exists():
533
- temp_vtp.rename(vtp)
544
+ shutil.move(temp_vtp, vtp)
534
545
  update_lock(False)
535
546
  return False
536
547
  if verify:
@@ -538,7 +549,9 @@ def init_venv(
538
549
  verified_venvs.add(venv)
539
550
 
540
551
  if rename_vtp and temp_vtp.exists():
541
- temp_vtp.rename(vtp)
552
+ if debug:
553
+ print(f"Cleanup: move '{temp_vtp}' back to '{vtp}'.")
554
+ shutil.move(temp_vtp, vtp)
542
555
 
543
556
  update_lock(False)
544
557
  return True
@@ -653,10 +666,10 @@ def venv_exists(venv: Union[str, None], debug: bool = False) -> bool:
653
666
 
654
667
 
655
668
  def venv_target_path(
656
- venv: Union[str, None],
657
- allow_nonexistent: bool = False,
658
- debug: bool = False,
659
- ) -> 'pathlib.Path':
669
+ venv: Union[str, None],
670
+ allow_nonexistent: bool = False,
671
+ debug: bool = False,
672
+ ) -> 'pathlib.Path':
660
673
  """
661
674
  Return a virtual environment's site-package path.
662
675
 
@@ -673,7 +686,11 @@ def venv_target_path(
673
686
  The `pathlib.Path` object for the virtual environment's path.
674
687
 
675
688
  """
676
- import os, sys, platform, pathlib, site
689
+ import os
690
+ import sys
691
+ import platform
692
+ import pathlib
693
+ import site
677
694
  from meerschaum.config._paths import VIRTENV_RESOURCES_PATH
678
695
  from meerschaum.config.static import STATIC_CONFIG
679
696
 
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: meerschaum
3
- Version: 2.7.3
3
+ Version: 2.7.5
4
4
  Summary: Sync Time-Series Pipes with Meerschaum
5
5
  Home-page: https://meerschaum.io
6
6
  Author: Bennett Meares
@@ -6,7 +6,6 @@ Test Meerschaum jobs.
6
6
  """
7
7
 
8
8
  import time
9
- import pytest
10
9
 
11
10
  import meerschaum as mrsm
12
11
 
@@ -16,18 +15,16 @@ def test_create_job():
16
15
  Test creating and running a new job.
17
16
  """
18
17
  sysargs = ['show', 'version', ':', '--loop', '--min-seconds', '0.1']
19
- job = mrsm.Job(
20
- 'test', sysargs, executor_keys='local'
21
- )
18
+ job = mrsm.Job('test', sysargs, executor_keys='local')
22
19
  job.delete()
23
20
  success, msg = job.start()
24
21
  assert success, msg
25
22
 
26
- duration = 4.0
27
- time.sleep(duration)
23
+ time.sleep(4.0)
28
24
 
29
25
  success, msg = job.stop()
30
26
  assert success, msg
27
+ time.sleep(1.0)
31
28
 
32
29
  success, msg = job.result
33
30
  assert success, msg