singlestoredb 1.7.0__py3-none-any.whl → 1.7.2__py3-none-any.whl

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.

Potentially problematic release.


This version of singlestoredb might be problematic. Click here for more details.

singlestoredb/__init__.py CHANGED
@@ -13,7 +13,7 @@ Examples
13
13
 
14
14
  """
15
15
 
16
- __version__ = '1.7.0'
16
+ __version__ = '1.7.2'
17
17
 
18
18
  from typing import Any
19
19
 
@@ -1,8 +1,9 @@
1
1
  import asyncio
2
+ import textwrap
2
3
  import typing
3
- import urllib.parse
4
4
 
5
5
  from ._config import AppConfig
6
+ from ._connection_info import ConnectionInfo
6
7
  from ._process import kill_process_by_port
7
8
 
8
9
  if typing.TYPE_CHECKING:
@@ -17,8 +18,7 @@ async def run_function_app(
17
18
  app: 'FastAPI',
18
19
  log_level: str = 'error',
19
20
  kill_existing_app_server: bool = True,
20
- ) -> None:
21
-
21
+ ) -> ConnectionInfo:
22
22
  global _running_server
23
23
  from ._uvicorn_util import AwaitableUvicornServer
24
24
 
@@ -52,8 +52,7 @@ async def run_function_app(
52
52
  def ping() -> str:
53
53
  return 'Success!'
54
54
 
55
- base_path = urllib.parse.urlparse(app_config.url).path
56
- app.root_path = base_path
55
+ app.root_path = app_config.base_path
57
56
 
58
57
  config = uvicorn.Config(
59
58
  app,
@@ -66,5 +65,26 @@ async def run_function_app(
66
65
  asyncio.create_task(_running_server.serve())
67
66
  await _running_server.wait_for_startup()
68
67
 
68
+ connection_info = ConnectionInfo(app_config.base_url, app_config.token)
69
+
69
70
  if app_config.running_interactively:
70
- print(f'Cloud function available at {app_config.url}')
71
+ if app_config.is_gateway_enabled:
72
+ print(
73
+ 'Cloud function available at '
74
+ f'{app_config.base_url}docs?authToken={app_config.token}',
75
+ )
76
+ else:
77
+ curl_header = f'-H "Authorization: Bearer {app_config.token}"'
78
+ curl_example = f'curl "{app_config.base_url}" {curl_header}'
79
+ print(
80
+ textwrap.dedent(f"""
81
+ Cloud function available at {app_config.base_url}
82
+
83
+ Auth Token: {app_config.token}
84
+
85
+ Curl example: {curl_example}
86
+
87
+ """).strip(),
88
+ )
89
+
90
+ return connection_info
@@ -1,33 +1,66 @@
1
1
  import os
2
2
  from dataclasses import dataclass
3
+ from typing import Optional
3
4
 
4
5
 
5
6
  @dataclass
6
7
  class AppConfig:
7
8
  listen_port: int
8
- url: str
9
+ base_url: str
10
+ base_path: str
11
+ app_token: Optional[str]
12
+ user_token: Optional[str]
9
13
  running_interactively: bool
14
+ is_gateway_enabled: bool
10
15
 
11
- @classmethod
12
- def from_env(cls) -> 'AppConfig':
13
- port = os.environ.get('SINGLESTOREDB_APP_LISTEN_PORT')
14
- if port is None:
15
- raise RuntimeError(
16
- 'Missing SINGLESTOREDB_APP_LISTEN_PORT environment variable. '
17
- 'Is the code running outside SingleStoreDB notebook environment?',
18
- )
19
- url = os.environ.get('SINGLESTOREDB_APP_URL')
20
- if url is None:
16
+ @staticmethod
17
+ def _read_variable(name: str) -> str:
18
+ value = os.environ.get(name)
19
+ if value is None:
21
20
  raise RuntimeError(
22
- 'Missing SINGLESTOREDB_APP_URL environment variable. '
21
+ f'Missing {name} environment variable. '
23
22
  'Is the code running outside SingleStoreDB notebook environment?',
24
23
  )
24
+ return value
25
+
26
+ @classmethod
27
+ def from_env(cls) -> 'AppConfig':
28
+ port = cls._read_variable('SINGLESTOREDB_APP_LISTEN_PORT')
29
+ base_url = cls._read_variable('SINGLESTOREDB_APP_BASE_URL')
30
+ base_path = cls._read_variable('SINGLESTOREDB_APP_BASE_PATH')
25
31
 
26
32
  workload_type = os.environ.get('SINGLESTOREDB_WORKLOAD_TYPE')
27
33
  running_interactively = workload_type == 'InteractiveNotebook'
28
34
 
35
+ is_gateway_enabled = 'SINGLESTOREDB_NOVA_GATEWAY_ENDPOINT' in os.environ
36
+
37
+ app_token = os.environ.get('SINGLESTOREDB_APP_TOKEN')
38
+ user_token = os.environ.get('SINGLESTOREDB_USER_TOKEN')
39
+
40
+ # Make sure the required variables are present
41
+ # and present useful error message if not
42
+ if running_interactively:
43
+ if is_gateway_enabled:
44
+ app_token = cls._read_variable('SINGLESTOREDB_APP_TOKEN')
45
+ else:
46
+ user_token = cls._read_variable('SINGLESTOREDB_USER_TOKEN')
47
+
29
48
  return cls(
30
49
  listen_port=int(port),
31
- url=url,
50
+ base_url=base_url,
51
+ base_path=base_path,
52
+ app_token=app_token,
53
+ user_token=user_token,
32
54
  running_interactively=running_interactively,
55
+ is_gateway_enabled=is_gateway_enabled,
33
56
  )
57
+
58
+ @property
59
+ def token(self) -> Optional[str]:
60
+ """
61
+ Returns None if running non-interactively
62
+ """
63
+ if self.is_gateway_enabled:
64
+ return self.app_token
65
+ else:
66
+ return self.user_token
@@ -0,0 +1,10 @@
1
+ from dataclasses import dataclass
2
+ from typing import Optional
3
+
4
+
5
+ @dataclass
6
+ class ConnectionInfo:
7
+ url: str
8
+
9
+ # Only present in interactive mode
10
+ token: Optional[str]
@@ -1,51 +1,47 @@
1
1
  import typing
2
- import urllib.parse
3
2
 
4
3
  from ._config import AppConfig
5
4
  from ._process import kill_process_by_port
5
+ from ._stdout_supress import StdoutSuppressor
6
+ from singlestoredb.apps._connection_info import ConnectionInfo
6
7
 
7
8
  if typing.TYPE_CHECKING:
8
- from plotly.graph_objs import Figure
9
+ from dash import Dash
9
10
 
10
11
 
11
- def run_dashboard_app(
12
- figure: 'Figure',
12
+ async def run_dashboard_app(
13
+ app: 'Dash',
13
14
  debug: bool = False,
14
15
  kill_existing_app_server: bool = True,
15
- ) -> None:
16
+ ) -> ConnectionInfo:
16
17
  try:
17
- import dash
18
+ from dash import Dash
18
19
  except ImportError:
19
20
  raise ImportError('package dash is required to run dashboards')
20
21
 
21
- try:
22
- from plotly.graph_objs import Figure
23
- except ImportError:
24
- raise ImportError('package dash is required to run dashboards')
25
-
26
- if not isinstance(figure, Figure):
27
- raise TypeError('figure is not an instance of plotly Figure')
22
+ if not isinstance(app, Dash):
23
+ raise TypeError('app is not an instance of Dash App')
28
24
 
29
25
  app_config = AppConfig.from_env()
30
26
 
31
27
  if kill_existing_app_server:
32
28
  kill_process_by_port(app_config.listen_port)
33
29
 
34
- base_path = urllib.parse.urlparse(app_config.url).path
35
-
36
- app = dash.Dash(requests_pathname_prefix=base_path)
37
- app.layout = dash.html.Div(
38
- [
39
- dash.dcc.Graph(figure=figure),
40
- ],
41
- )
42
-
43
- app.run(
44
- host='0.0.0.0',
45
- debug=debug,
46
- port=str(app_config.listen_port),
47
- jupyter_mode='external',
48
- )
30
+ if app.config.requests_pathname_prefix is None or \
31
+ app.config.requests_pathname_prefix != app_config.base_path:
32
+ raise RuntimeError('''
33
+ requests_pathname_prefix of the Dash App is invalid. Please set
34
+ requests_pathname_prefix=os.environ['SINGLESTOREDB_APP_BASE_PATH']
35
+ while initializing the Dash App and retry''')
36
+
37
+ with StdoutSuppressor():
38
+ app.run(
39
+ host='0.0.0.0',
40
+ debug=debug,
41
+ port=str(app_config.listen_port),
42
+ jupyter_mode='external',
43
+ )
49
44
 
50
45
  if app_config.running_interactively:
51
- print(f'Dash app available at {app_config.url}')
46
+ print(f'Dash app available at {app_config.base_url}?authToken={app_config.token}')
47
+ return ConnectionInfo(app_config.base_url, app_config.token)
@@ -0,0 +1,30 @@
1
+ import io
2
+ import sys
3
+ from typing import Optional
4
+
5
+
6
+ class StdoutSuppressor:
7
+ """
8
+ Supresses the stdout for code executed within the context.
9
+ This should not be used for asynchronous or threaded executions.
10
+
11
+ ```py
12
+ with Supressor():
13
+ print("This won't be printed")
14
+ ```
15
+
16
+ """
17
+
18
+ def __enter__(self) -> None:
19
+ self.stdout = sys.stdout
20
+ self.buffer = io.StringIO()
21
+ sys.stdout = self.buffer
22
+
23
+ def __exit__(
24
+ self,
25
+ exc_type: Optional[object],
26
+ exc_value: Optional[Exception],
27
+ exc_traceback: Optional[str],
28
+ ) -> None:
29
+ del self.buffer
30
+ sys.stdout = self.stdout
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: singlestoredb
3
- Version: 1.7.0
3
+ Version: 1.7.2
4
4
  Summary: Interface to the SingleStoreDB database and workspace management APIs
5
5
  Home-page: https://github.com/singlestore-labs/singlestoredb-python
6
6
  Author: SingleStore
@@ -1,4 +1,4 @@
1
- singlestoredb/__init__.py,sha256=dYAymV6mtmpkfZMMdLoAUjKcpN7Lh7jEOdNeFM3I7jM,1634
1
+ singlestoredb/__init__.py,sha256=UjufrohnvC56tM8JigZbe1K0NDetQIji_savOS-HmXc,1634
2
2
  singlestoredb/auth.py,sha256=u8D9tpKzrqa4ssaHjyZnGDX1q8XBpGtuoOkTkSv7B28,7599
3
3
  singlestoredb/config.py,sha256=NtONv4Etpraoy1nenHqRAS08xHJZmho00J95uDjLxQM,12290
4
4
  singlestoredb/connection.py,sha256=x5lINBa9kB_GoEEeL2uUZi9G8pNwKuFtA1uqJirR6HI,45352
@@ -10,10 +10,12 @@ singlestoredb/ai/__init__.py,sha256=7Pubobzx5OlyepNo5DOOxWev1DUW9WFc9P6Qver2xpY,
10
10
  singlestoredb/ai/embeddings.py,sha256=3jghE4WMf7vy8RobhrMOLvMLnDNGbkPCF48B3fGM38U,746
11
11
  singlestoredb/alchemy/__init__.py,sha256=dXRThusYrs_9GjrhPOw0-vw94in_T8yY9jE7SGCqiQk,2523
12
12
  singlestoredb/apps/__init__.py,sha256=uuEH2WZ1ROpmkMBBdz1tSkQSdYR9blXXU2nn7E5P4qQ,118
13
- singlestoredb/apps/_cloud_functions.py,sha256=qIykvR7TuTXvrNMOi25ejklQx79CpScC3tV49cejJR0,2036
14
- singlestoredb/apps/_config.py,sha256=kCkuEnW8rXvGvG48o4QY35PSiIm4AyltqJn_bNoHmDI,1056
15
- singlestoredb/apps/_dashboards.py,sha256=dSyc2NO8ocni6HC_ooruokiJgzEYUBLFg4ybS1FD9mM,1272
13
+ singlestoredb/apps/_cloud_functions.py,sha256=NJJu0uJsK9TjY3yZjgftpFPR-ga-FrOyaiDD4jWFCtE,2704
14
+ singlestoredb/apps/_config.py,sha256=w21kH0jMJ0_cP_VgAxHhKiLW5Iyrr7xzUIxRmfe_fqs,2118
15
+ singlestoredb/apps/_connection_info.py,sha256=gQPYzJrBQUEH76zVTkxJ7FAypNoN2T7GYHVOSgJ7Q8Q,175
16
+ singlestoredb/apps/_dashboards.py,sha256=_03fI-GJannamA5lxLvIoC6Mim-H1jTRuI8-dw_P--k,1474
16
17
  singlestoredb/apps/_process.py,sha256=G37fk6bzIxzhfEqp2aJBk3JCij-T2HFtTd078k5Xq9I,944
18
+ singlestoredb/apps/_stdout_supress.py,sha256=8s9zMIIRPpeu44yluJFc_0VueAxZDmr9QVGT6TGiFeY,659
17
19
  singlestoredb/apps/_uvicorn_util.py,sha256=rEK4nEmq5hbpRgsmK16UVlxe2DyQSq7C5w5WZSp0kX8,962
18
20
  singlestoredb/functions/__init__.py,sha256=WL1LqgMTdnGOse3tQqmD-HH8TdfCPS89GNO7hO0v_aw,41
19
21
  singlestoredb/functions/decorator.py,sha256=H12MUeBw8VOppx6esntaR43ukeIffbnAr716CBpYJ4g,5193
@@ -122,9 +124,9 @@ singlestoredb/utils/events.py,sha256=9IB84T3pKQjs7aaoSSJCw7soNngnhoTDWIC52M51R9Y
122
124
  singlestoredb/utils/mogrify.py,sha256=-a56IF70U6CkfadeaZgfjRSVsAD3PuqRrzPpjZlgbwY,4050
123
125
  singlestoredb/utils/results.py,sha256=bJtaUaDiFq26IsPAKZ2FHGB7csMn94EAxLKrP4HaEEA,15277
124
126
  singlestoredb/utils/xdict.py,sha256=S9HKgrPrnu_6b7iOwa2KrW8CmU1Uqx0BWdEyogFzWbE,12896
125
- singlestoredb-1.7.0.dist-info/LICENSE,sha256=Mlq78idURT-9G026aMYswwwnnrLcgzTLuXeAs5hjDLM,11341
126
- singlestoredb-1.7.0.dist-info/METADATA,sha256=hut54uSeTSVJL5_3nTXZ5RaHNedRj7C1UU4YM0uOLG8,5557
127
- singlestoredb-1.7.0.dist-info/WHEEL,sha256=eOLhNAGa2EW3wWl_TU484h7q1UNgy0JXjjoqKoxAAQc,92
128
- singlestoredb-1.7.0.dist-info/entry_points.txt,sha256=bSLaTWB5zGjpVYPAaI46MkkDup0su-eb3uAhCNYuRV0,48
129
- singlestoredb-1.7.0.dist-info/top_level.txt,sha256=eet8bVPNRqiGeY0PrO5ERH2UpamwlrKHEQCffz4dOh8,14
130
- singlestoredb-1.7.0.dist-info/RECORD,,
127
+ singlestoredb-1.7.2.dist-info/LICENSE,sha256=Mlq78idURT-9G026aMYswwwnnrLcgzTLuXeAs5hjDLM,11341
128
+ singlestoredb-1.7.2.dist-info/METADATA,sha256=k_cNa90181RAA4XGHdWKHlQdl-9A-shPl86g7d3dJV0,5557
129
+ singlestoredb-1.7.2.dist-info/WHEEL,sha256=eOLhNAGa2EW3wWl_TU484h7q1UNgy0JXjjoqKoxAAQc,92
130
+ singlestoredb-1.7.2.dist-info/entry_points.txt,sha256=bSLaTWB5zGjpVYPAaI46MkkDup0su-eb3uAhCNYuRV0,48
131
+ singlestoredb-1.7.2.dist-info/top_level.txt,sha256=eet8bVPNRqiGeY0PrO5ERH2UpamwlrKHEQCffz4dOh8,14
132
+ singlestoredb-1.7.2.dist-info/RECORD,,