amsdal 0.5.19__cp312-cp312-macosx_10_13_universal2.whl → 0.5.21__cp312-cp312-macosx_10_13_universal2.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 amsdal might be problematic. Click here for more details.

Files changed (62) hide show
  1. amsdal/__about__.py +1 -1
  2. amsdal/cloud/__init__.cpython-312-darwin.so +0 -0
  3. amsdal/cloud/client.cpython-312-darwin.so +0 -0
  4. amsdal/cloud/constants.cpython-312-darwin.so +0 -0
  5. amsdal/cloud/enums.cpython-312-darwin.so +0 -0
  6. amsdal/cloud/models/__init__.cpython-312-darwin.so +0 -0
  7. amsdal/cloud/models/base.cpython-312-darwin.so +0 -0
  8. amsdal/cloud/services/__init__.cpython-312-darwin.so +0 -0
  9. amsdal/cloud/services/actions/__init__.cpython-312-darwin.so +0 -0
  10. amsdal/cloud/services/actions/add_allowlist_ip.cpython-312-darwin.so +0 -0
  11. amsdal/cloud/services/actions/add_basic_auth.cpython-312-darwin.so +0 -0
  12. amsdal/cloud/services/actions/add_dependency.cpython-312-darwin.so +0 -0
  13. amsdal/cloud/services/actions/add_secret.cpython-312-darwin.so +0 -0
  14. amsdal/cloud/services/actions/base.cpython-312-darwin.so +0 -0
  15. amsdal/cloud/services/actions/create_deploy.cpython-312-darwin.so +0 -0
  16. amsdal/cloud/services/actions/create_env.cpython-312-darwin.so +0 -0
  17. amsdal/cloud/services/actions/create_session.cpython-312-darwin.so +0 -0
  18. amsdal/cloud/services/actions/delete_allowlist_ip.cpython-312-darwin.so +0 -0
  19. amsdal/cloud/services/actions/delete_basic_auth.cpython-312-darwin.so +0 -0
  20. amsdal/cloud/services/actions/delete_dependency.cpython-312-darwin.so +0 -0
  21. amsdal/cloud/services/actions/delete_env.cpython-312-darwin.so +0 -0
  22. amsdal/cloud/services/actions/delete_secret.cpython-312-darwin.so +0 -0
  23. amsdal/cloud/services/actions/destroy_deploy.cpython-312-darwin.so +0 -0
  24. amsdal/cloud/services/actions/expose_db.cpython-312-darwin.so +0 -0
  25. amsdal/cloud/services/actions/get_basic_auth_credentials.cpython-312-darwin.so +0 -0
  26. amsdal/cloud/services/actions/get_monitoring_info.cpython-312-darwin.so +0 -0
  27. amsdal/cloud/services/actions/list_dependencies.cpython-312-darwin.so +0 -0
  28. amsdal/cloud/services/actions/list_deploys.cpython-312-darwin.so +0 -0
  29. amsdal/cloud/services/actions/list_envs.cpython-312-darwin.so +0 -0
  30. amsdal/cloud/services/actions/list_secrets.cpython-312-darwin.so +0 -0
  31. amsdal/cloud/services/actions/manager.cpython-312-darwin.so +0 -0
  32. amsdal/cloud/services/actions/signup_action.cpython-312-darwin.so +0 -0
  33. amsdal/cloud/services/actions/update_deploy.cpython-312-darwin.so +0 -0
  34. amsdal/cloud/services/auth/__init__.cpython-312-darwin.so +0 -0
  35. amsdal/cloud/services/auth/base.cpython-312-darwin.so +0 -0
  36. amsdal/cloud/services/auth/credentials.cpython-312-darwin.so +0 -0
  37. amsdal/cloud/services/auth/manager.cpython-312-darwin.so +0 -0
  38. amsdal/cloud/services/auth/signup_service.cpython-312-darwin.so +0 -0
  39. amsdal/cloud/services/auth/token.cpython-312-darwin.so +0 -0
  40. amsdal/contrib/__init__.cpython-312-darwin.so +0 -0
  41. amsdal/contrib/frontend_configs/migrations/0002_add_button_and_invoke_actions.py +314 -0
  42. amsdal/contrib/frontend_configs/models/frontend_config_control_action.py +57 -1
  43. amsdal/contrib/frontend_configs/models/frontend_control_config.py +10 -1
  44. amsdal/fixtures/__init__.cpython-312-darwin.so +0 -0
  45. amsdal/fixtures/manager.cpython-312-darwin.so +0 -0
  46. amsdal/fixtures/utils.cpython-312-darwin.so +0 -0
  47. amsdal/manager.cpython-312-darwin.so +0 -0
  48. amsdal/mixins/__init__.cpython-312-darwin.so +0 -0
  49. amsdal/mixins/class_versions_mixin.cpython-312-darwin.so +0 -0
  50. amsdal/schemas/manager.cpython-312-darwin.so +0 -0
  51. amsdal/services/__init__.py +11 -0
  52. amsdal/services/external_connections.py +262 -0
  53. amsdal/services/external_model_generator.py +350 -0
  54. amsdal/services/transaction_execution.cpython-312-darwin.so +0 -0
  55. {amsdal-0.5.19.dist-info → amsdal-0.5.21.dist-info}/METADATA +1 -1
  56. {amsdal-0.5.19.dist-info → amsdal-0.5.21.dist-info}/RECORD +59 -58
  57. amsdal/services/__init__.cpython-312-darwin.so +0 -0
  58. amsdal/services/external_connections.cpython-312-darwin.so +0 -0
  59. amsdal/services/external_model_generator.cpython-312-darwin.so +0 -0
  60. {amsdal-0.5.19.dist-info → amsdal-0.5.21.dist-info}/WHEEL +0 -0
  61. {amsdal-0.5.19.dist-info → amsdal-0.5.21.dist-info}/licenses/LICENSE.txt +0 -0
  62. {amsdal-0.5.19.dist-info → amsdal-0.5.21.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,262 @@
1
+ """
2
+ External Connection Manager for accessing external services and databases.
3
+
4
+ This module provides a high-level interface for working with external connections
5
+ such as read-only databases, email services, storage services, etc.
6
+ """
7
+
8
+ from typing import Any
9
+ from typing import TypeVar
10
+
11
+ from amsdal_data.application import AsyncDataApplication
12
+ from amsdal_data.application import DataApplication
13
+ from amsdal_utils.utils.singleton import Singleton
14
+
15
+ T = TypeVar('T')
16
+
17
+
18
+ class ExternalConnectionManager(metaclass=Singleton):
19
+ """
20
+ Manager for accessing external service connections.
21
+
22
+ Provides a convenient interface to access external connections configured
23
+ in the application, such as read-only databases, email services, etc.
24
+
25
+ Example usage:
26
+ manager = ExternalConnectionManager()
27
+
28
+ # Get read-only database connection
29
+ external_db = manager.get_connection('external_users_db')
30
+ rows = external_db.fetch_all('SELECT * FROM users WHERE active = 1')
31
+
32
+ # Get email service
33
+ email = manager.get_connection('email_service')
34
+ email.send_email(...)
35
+ """
36
+
37
+ def __init__(self) -> None:
38
+ self._data_application: DataApplication | None = None
39
+ self._async_data_application: AsyncDataApplication | None = None
40
+
41
+ def setup(
42
+ self,
43
+ data_application: DataApplication | None = None,
44
+ async_data_application: AsyncDataApplication | None = None,
45
+ ) -> None:
46
+ """
47
+ Set up the manager with the data application instance.
48
+
49
+ Args:
50
+ data_application: Sync DataApplication instance
51
+ async_data_application: Async DataApplication instance
52
+ """
53
+ self._data_application = data_application
54
+ self._async_data_application = async_data_application
55
+
56
+ def get_connection(self, name: str) -> Any:
57
+ """
58
+ Get an external service connection by name.
59
+
60
+ Args:
61
+ name: Name of the external connection (as configured in resources)
62
+
63
+ Returns:
64
+ The external connection object
65
+
66
+ Raises:
67
+ RuntimeError: If manager is not set up
68
+ KeyError: If connection not found
69
+ """
70
+ if self._data_application is None and self._async_data_application is None:
71
+ msg = 'ExternalConnectionManager not set up. Call setup() first.'
72
+ raise RuntimeError(msg)
73
+
74
+ app = self._data_application or self._async_data_application
75
+ if app is None: # Shouldn't happen due to check above, but satisfy mypy
76
+ msg = 'No data application available'
77
+ raise RuntimeError(msg)
78
+ return app.get_external_service_connection(name)
79
+
80
+ def has_connection(self, name: str) -> bool:
81
+ """
82
+ Check if an external connection exists.
83
+
84
+ Args:
85
+ name: Name of the external connection
86
+
87
+ Returns:
88
+ bool: True if connection exists, False otherwise
89
+ """
90
+ if self._data_application is None and self._async_data_application is None:
91
+ return False
92
+
93
+ try:
94
+ self.get_connection(name)
95
+ return True
96
+ except KeyError:
97
+ return False
98
+
99
+ def list_connections(self) -> list[str]:
100
+ """
101
+ List all available external connection names.
102
+
103
+ Returns:
104
+ list[str]: List of connection names
105
+ """
106
+ if self._data_application:
107
+ return list(self._data_application._external_service_connections.keys()) # noqa: SLF001
108
+ if self._async_data_application:
109
+ return list(self._async_data_application._external_service_connections.keys()) # noqa: SLF001
110
+ return []
111
+
112
+
113
+ class ExternalDatabaseReader:
114
+ """
115
+ Helper class for reading from external read-only databases.
116
+
117
+ Provides a convenient interface for querying external databases
118
+ with common patterns like filtering, mapping results, etc.
119
+
120
+ Example usage:
121
+ reader = ExternalDatabaseReader('external_users_db')
122
+
123
+ # Fetch all users
124
+ users = reader.fetch_all('SELECT * FROM users')
125
+
126
+ # Fetch with parameters
127
+ active_users = reader.fetch_all(
128
+ 'SELECT * FROM users WHERE active = ?',
129
+ (1,)
130
+ )
131
+
132
+ # Fetch one record
133
+ user = reader.fetch_one('SELECT * FROM users WHERE id = ?', (user_id,))
134
+
135
+ # Get as dictionaries
136
+ user_dicts = reader.fetch_all_as_dicts('SELECT * FROM users LIMIT 10')
137
+ """
138
+
139
+ def __init__(self, connection_name: str):
140
+ """
141
+ Initialize the reader with a connection name.
142
+
143
+ Args:
144
+ connection_name: Name of the external database connection
145
+ """
146
+ self.connection_name = connection_name
147
+ self._manager = ExternalConnectionManager()
148
+
149
+ @property
150
+ def connection(self) -> Any:
151
+ """Get the underlying connection object."""
152
+ return self._manager.get_connection(self.connection_name)
153
+
154
+ def fetch_all(self, query: str, parameters: tuple[Any, ...] | None = None) -> list[Any]:
155
+ """
156
+ Execute query and fetch all results.
157
+
158
+ Args:
159
+ query: SQL query to execute
160
+ parameters: Query parameters (optional)
161
+
162
+ Returns:
163
+ list: List of result rows
164
+ """
165
+ return self.connection.fetch_all(query, parameters)
166
+
167
+ def fetch_one(self, query: str, parameters: tuple[Any, ...] | None = None) -> Any | None:
168
+ """
169
+ Execute query and fetch one result.
170
+
171
+ Args:
172
+ query: SQL query to execute
173
+ parameters: Query parameters (optional)
174
+
175
+ Returns:
176
+ Single result row or None
177
+ """
178
+ return self.connection.fetch_one(query, parameters)
179
+
180
+ def fetch_all_as_dicts(self, query: str, parameters: tuple[Any, ...] | None = None) -> list[dict[str, Any]]:
181
+ """
182
+ Execute query and fetch all results as dictionaries.
183
+
184
+ Args:
185
+ query: SQL query to execute
186
+ parameters: Query parameters (optional)
187
+
188
+ Returns:
189
+ list[dict]: List of result dictionaries
190
+ """
191
+ rows = self.fetch_all(query, parameters)
192
+ return [dict(row) for row in rows]
193
+
194
+ def fetch_one_as_dict(self, query: str, parameters: tuple[Any, ...] | None = None) -> dict[str, Any] | None:
195
+ """
196
+ Execute query and fetch one result as dictionary.
197
+
198
+ Args:
199
+ query: SQL query to execute
200
+ parameters: Query parameters (optional)
201
+
202
+ Returns:
203
+ dict | None: Result dictionary or None
204
+ """
205
+ row = self.fetch_one(query, parameters)
206
+ return dict(row) if row else None
207
+
208
+ def get_table_names(self) -> list[str]:
209
+ """
210
+ Get list of all tables in the database.
211
+
212
+ Returns:
213
+ list[str]: List of table names
214
+ """
215
+ return self.connection.get_table_names()
216
+
217
+ def get_table_schema(self, table_name: str) -> list[dict[str, Any]]:
218
+ """
219
+ Get schema information for a table.
220
+
221
+ Args:
222
+ table_name: Name of the table
223
+
224
+ Returns:
225
+ list[dict]: List of column information dictionaries
226
+ """
227
+ return self.connection.get_table_schema(table_name)
228
+
229
+ def count(self, table: str, where_clause: str = '', parameters: tuple[Any, ...] | None = None) -> int:
230
+ """
231
+ Count rows in a table.
232
+
233
+ Args:
234
+ table: Table name
235
+ where_clause: Optional WHERE clause (without WHERE keyword)
236
+ parameters: Query parameters for WHERE clause
237
+
238
+ Returns:
239
+ int: Number of rows
240
+ """
241
+ query = f'SELECT COUNT(*) as count FROM {table}' # noqa: S608
242
+ if where_clause:
243
+ query += f' WHERE {where_clause}'
244
+
245
+ result = self.fetch_one(query, parameters)
246
+ return result['count'] if result else 0
247
+
248
+ def exists(self, table: str, where_clause: str, parameters: tuple[Any, ...]) -> bool:
249
+ """
250
+ Check if a record exists.
251
+
252
+ Args:
253
+ table: Table name
254
+ where_clause: WHERE clause (without WHERE keyword)
255
+ parameters: Query parameters
256
+
257
+ Returns:
258
+ bool: True if record exists, False otherwise
259
+ """
260
+ query = f'SELECT 1 FROM {table} WHERE {where_clause} LIMIT 1' # noqa: S608
261
+ result = self.fetch_one(query, parameters)
262
+ return result is not None
@@ -0,0 +1,350 @@
1
+ """
2
+ External Model Generator Service.
3
+
4
+ Generates ExternalModel classes from external connection schemas.
5
+ This allows runtime model generation from external databases without
6
+ manual model definition.
7
+ """
8
+
9
+ from typing import Any
10
+ from typing import cast
11
+
12
+ from amsdal_data.connections.external.base import SchemaIntrospectionProtocol
13
+ from amsdal_models.classes.external_model import ExternalModel
14
+ from amsdal_models.utils.schema_converter import ExternalSchemaConverter
15
+ from amsdal_utils.schemas.schema import ObjectSchema
16
+
17
+ from amsdal.services.external_connections import ExternalConnectionManager
18
+
19
+
20
+ class ExternalModelGenerator:
21
+ """
22
+ Service for generating ExternalModel classes from external connections.
23
+
24
+ This service introspects external database schemas and generates
25
+ corresponding ExternalModel classes that can be used immediately
26
+ for querying the external data.
27
+
28
+ Features:
29
+ - Automatic schema introspection
30
+ - Type mapping (SQL types -> Python types)
31
+ - Primary key detection
32
+ - In-memory model class generation
33
+ - No lakehouse schema creation
34
+
35
+ Example usage:
36
+ # Generate models for a single table
37
+ generator = ExternalModelGenerator()
38
+ User = generator.generate_model('external_db', 'users')
39
+
40
+ # Now use the generated model
41
+ users = User.objects.filter(active=True).execute()
42
+
43
+ # Generate models for all tables
44
+ models = generator.generate_models_for_connection('external_db')
45
+ User = models['User']
46
+ Post = models['Post']
47
+ """
48
+
49
+ def __init__(self) -> None:
50
+ self._connection_manager = ExternalConnectionManager()
51
+ self._schema_converter = ExternalSchemaConverter()
52
+
53
+ def generate_model(
54
+ self,
55
+ connection_name: str,
56
+ table_name: str,
57
+ model_name: str | None = None,
58
+ ) -> type[ExternalModel]:
59
+ """
60
+ Generate an ExternalModel class for a specific table.
61
+
62
+ Args:
63
+ connection_name: Name of the external connection
64
+ table_name: Name of the table to generate model for
65
+ model_name: Optional custom model name (defaults to classified table name)
66
+
67
+ Returns:
68
+ type[ExternalModel]: Generated model class ready to use
69
+
70
+ Raises:
71
+ ValueError: If connection doesn't support schema introspection
72
+ ConnectionError: If connection is not available
73
+ RuntimeError: If model generation fails
74
+
75
+ Example:
76
+ generator = ExternalModelGenerator()
77
+ User = generator.generate_model('external_db', 'users')
78
+
79
+ # Query using the generated model
80
+ active_users = User.objects.filter(active=True).execute()
81
+ """
82
+ # Get the connection
83
+ connection = self._connection_manager.get_connection(connection_name)
84
+
85
+ # Check if connection supports schema introspection
86
+ if not isinstance(connection, SchemaIntrospectionProtocol): # type: ignore[misc]
87
+ msg = (
88
+ f"Connection '{connection_name}' does not support schema introspection. "
89
+ f'Connection type: {type(connection).__name__}'
90
+ )
91
+ raise ValueError(msg)
92
+
93
+ # Get table schema
94
+ table_schema = connection.get_table_schema(table_name)
95
+
96
+ # Convert to ObjectSchema
97
+ # Detect connection type and use appropriate converter
98
+ object_schema = self._convert_schema(
99
+ connection=connection,
100
+ table_name=table_name,
101
+ table_schema=table_schema,
102
+ connection_name=connection_name,
103
+ )
104
+
105
+ # Generate model class from ObjectSchema
106
+ model_class = self._create_model_class(object_schema, model_name)
107
+
108
+ return model_class
109
+
110
+ def generate_models_for_connection(
111
+ self,
112
+ connection_name: str,
113
+ table_names: list[str] | None = None,
114
+ ) -> dict[str, type[ExternalModel]]:
115
+ """
116
+ Generate ExternalModel classes for all tables in a connection.
117
+
118
+ Args:
119
+ connection_name: Name of the external connection
120
+ table_names: Optional list of specific tables to generate models for.
121
+ If None, generates models for all tables.
122
+
123
+ Returns:
124
+ dict[str, type[ExternalModel]]: Dictionary mapping model names to model classes
125
+
126
+ Raises:
127
+ ValueError: If connection doesn't support schema introspection
128
+ ConnectionError: If connection is not available
129
+
130
+ Example:
131
+ generator = ExternalModelGenerator()
132
+ models = generator.generate_models_for_connection('external_db')
133
+
134
+ # Access generated models
135
+ User = models['User']
136
+ Post = models['Post']
137
+ Comment = models['Comment']
138
+
139
+ # Or generate only specific tables
140
+ models = generator.generate_models_for_connection(
141
+ 'external_db',
142
+ table_names=['users', 'posts']
143
+ )
144
+ """
145
+ # Get the connection
146
+ connection = self._connection_manager.get_connection(connection_name)
147
+
148
+ # Check if connection supports schema introspection
149
+ if not isinstance(connection, SchemaIntrospectionProtocol): # type: ignore[misc]
150
+ msg = (
151
+ f"Connection '{connection_name}' does not support schema introspection. "
152
+ f'Connection type: {type(connection).__name__}'
153
+ )
154
+ raise ValueError(msg)
155
+
156
+ # Get list of tables
157
+ if table_names is None:
158
+ table_names = connection.get_table_names()
159
+
160
+ # Generate models for each table
161
+ models: dict[str, type[ExternalModel]] = {}
162
+ for table_name in table_names:
163
+ try:
164
+ model = self.generate_model(connection_name, table_name)
165
+ models[model.__name__] = model
166
+ except Exception as e:
167
+ # Log error but continue with other tables
168
+ print(f"Warning: Failed to generate model for table '{table_name}': {e}")
169
+ continue
170
+
171
+ return models
172
+
173
+ def _convert_schema(
174
+ self,
175
+ connection: Any,
176
+ table_name: str,
177
+ table_schema: list[dict[str, Any]],
178
+ connection_name: str,
179
+ ) -> ObjectSchema:
180
+ """
181
+ Convert raw table schema to ObjectSchema based on connection type.
182
+
183
+ Args:
184
+ connection: The connection object
185
+ table_name: Name of the table
186
+ table_schema: Raw schema data from connection
187
+ connection_name: Name of the connection
188
+
189
+ Returns:
190
+ ObjectSchema: Converted schema
191
+ """
192
+ # Detect connection type and use appropriate converter
193
+ connection_type = type(connection).__name__
194
+
195
+ if 'sqlite' in connection_type.lower():
196
+ return self._schema_converter.sqlite_schema_to_object_schema(
197
+ table_name=table_name,
198
+ columns=table_schema,
199
+ connection_name=connection_name,
200
+ )
201
+
202
+ # For other connection types, try to use generic converter
203
+ # First, try to detect the schema format
204
+ if table_schema and isinstance(table_schema[0], dict):
205
+ # Check if it's SQLite format (has 'cid', 'name', 'type', 'pk', etc.)
206
+ if all(key in table_schema[0] for key in ('cid', 'name', 'type')):
207
+ return self._schema_converter.sqlite_schema_to_object_schema(
208
+ table_name=table_name,
209
+ columns=table_schema,
210
+ connection_name=connection_name,
211
+ )
212
+
213
+ # Check if it's PostgreSQL format (has 'column_name', 'data_type', etc.)
214
+ if 'column_name' in table_schema[0] and 'data_type' in table_schema[0]:
215
+ return self._schema_converter.postgres_schema_to_object_schema(
216
+ table_name=table_name,
217
+ columns=table_schema,
218
+ connection_name=connection_name,
219
+ )
220
+
221
+ # Try generic converter with format normalization
222
+ normalized_columns = self._normalize_schema_format(table_schema)
223
+ return self._schema_converter.generic_schema_to_object_schema(
224
+ table_name=table_name,
225
+ columns=normalized_columns,
226
+ connection_name=connection_name,
227
+ )
228
+
229
+ msg = f'Unknown schema format for connection type: {connection_type}'
230
+ raise ValueError(msg)
231
+
232
+ def _normalize_schema_format(self, table_schema: list[dict[str, Any]]) -> list[dict[str, Any]]:
233
+ """
234
+ Normalize various schema formats to generic format.
235
+
236
+ Converts various schema formats to the format expected by generic_schema_to_object_schema:
237
+ {'name': str, 'type': str, 'nullable': bool, 'primary_key': bool, 'default': Any}
238
+ """
239
+ normalized = []
240
+
241
+ for column in table_schema:
242
+ # Try to extract name
243
+ name = column.get('name') or column.get('column_name') or column.get('field')
244
+
245
+ # Try to extract type
246
+ col_type = column.get('type') or column.get('data_type') or column.get('field_type') or 'TEXT'
247
+
248
+ # Try to extract nullable
249
+ nullable = True
250
+ if 'nullable' in column:
251
+ nullable = column['nullable']
252
+ elif 'is_nullable' in column:
253
+ nullable = column['is_nullable'] in (True, 'YES', 'yes', 1)
254
+ elif 'notnull' in column:
255
+ nullable = column['notnull'] in (False, 0)
256
+
257
+ # Try to extract primary key
258
+ pk = column.get('primary_key') or column.get('pk') or False
259
+ if isinstance(pk, int):
260
+ pk = pk > 0
261
+
262
+ # Try to extract default
263
+ default = column.get('default') or column.get('dflt_value') or column.get('column_default')
264
+
265
+ normalized.append(
266
+ {
267
+ 'name': name,
268
+ 'type': col_type,
269
+ 'nullable': nullable,
270
+ 'primary_key': pk,
271
+ 'default': default,
272
+ }
273
+ )
274
+
275
+ return normalized
276
+
277
+ def _create_model_class(
278
+ self,
279
+ object_schema: ObjectSchema,
280
+ custom_name: str | None = None,
281
+ ) -> type[ExternalModel]:
282
+ """
283
+ Create an ExternalModel class from ObjectSchema.
284
+
285
+ Args:
286
+ object_schema: The schema to create model from
287
+ custom_name: Optional custom model name
288
+
289
+ Returns:
290
+ type[ExternalModel]: Generated model class
291
+ """
292
+ # Extract model metadata from schema
293
+ model_name = custom_name or object_schema.title
294
+ table_name = cast(str, object_schema.__table_name__) # type: ignore[attr-defined]
295
+ connection_name = cast(str, object_schema.__connection__) # type: ignore[attr-defined]
296
+ pk_fields = getattr(object_schema, '__primary_key__', None)
297
+
298
+ # Build class attributes
299
+ class_attrs: dict[str, Any] = {
300
+ '__table_name__': table_name,
301
+ '__connection__': connection_name,
302
+ '__module__': __name__,
303
+ }
304
+
305
+ # Add primary key if present
306
+ if pk_fields:
307
+ # For composite keys, use list; for single key, use string
308
+ if len(pk_fields) == 1:
309
+ class_attrs['__primary_key__'] = pk_fields[0]
310
+ else:
311
+ class_attrs['__primary_key__'] = pk_fields
312
+
313
+ # Add field annotations from schema properties
314
+ annotations: dict[str, type] = {}
315
+ if object_schema.properties:
316
+ for field_name, field_def in object_schema.properties.items():
317
+ # Map CoreTypes to Python types for annotations
318
+ field_type = self._core_type_to_python_type(getattr(field_def, 'type', 'string'))
319
+ annotations[field_name] = field_type
320
+
321
+ class_attrs['__annotations__'] = annotations
322
+
323
+ # Create the model class dynamically
324
+ model_class = type(model_name, (ExternalModel,), class_attrs)
325
+
326
+ return cast(type[ExternalModel], model_class)
327
+
328
+ @staticmethod
329
+ def _core_type_to_python_type(core_type: str) -> type:
330
+ """
331
+ Convert CoreType string to Python type for annotations.
332
+
333
+ Args:
334
+ core_type: CoreType value (e.g., 'string', 'integer')
335
+
336
+ Returns:
337
+ type: Corresponding Python type
338
+ """
339
+ type_mapping = {
340
+ 'string': str,
341
+ 'integer': int,
342
+ 'number': float,
343
+ 'boolean': bool,
344
+ 'date': str, # Will be string representation
345
+ 'datetime': str, # Will be string representation
346
+ 'binary': bytes,
347
+ 'array': list,
348
+ 'dictionary': dict,
349
+ }
350
+ return type_mapping.get(core_type, str)
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: amsdal
3
- Version: 0.5.19
3
+ Version: 0.5.21
4
4
  Summary: AMSDAL
5
5
  License: AMSDAL End User License Agreement
6
6