airbyte-agent-shopify 0.1.24__tar.gz → 0.1.26__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (62) hide show
  1. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/AUTH.md +3 -0
  2. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/CHANGELOG.md +10 -0
  3. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/PKG-INFO +3 -3
  4. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/README.md +2 -2
  5. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/__init__.py +2 -2
  6. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/cloud_utils/client.py +72 -9
  7. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/schema/base.py +32 -1
  8. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/schema/extensions.py +19 -1
  9. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/schema/security.py +37 -0
  10. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/connector.py +8 -2
  11. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/pyproject.toml +1 -1
  12. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/.gitignore +0 -0
  13. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/REFERENCE.md +0 -0
  14. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/__init__.py +0 -0
  15. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/__init__.py +0 -0
  16. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/auth_strategies.py +0 -0
  17. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/auth_template.py +0 -0
  18. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/cloud_utils/__init__.py +0 -0
  19. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/connector_model_loader.py +0 -0
  20. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/constants.py +0 -0
  21. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/exceptions.py +0 -0
  22. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/executor/__init__.py +0 -0
  23. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/executor/hosted_executor.py +0 -0
  24. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/executor/local_executor.py +0 -0
  25. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/executor/models.py +0 -0
  26. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/extensions.py +0 -0
  27. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/http/__init__.py +0 -0
  28. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/http/adapters/__init__.py +0 -0
  29. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/http/adapters/httpx_adapter.py +0 -0
  30. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/http/config.py +0 -0
  31. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/http/exceptions.py +0 -0
  32. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/http/protocols.py +0 -0
  33. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/http/response.py +0 -0
  34. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/http_client.py +0 -0
  35. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/introspection.py +0 -0
  36. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/logging/__init__.py +0 -0
  37. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/logging/logger.py +0 -0
  38. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/logging/types.py +0 -0
  39. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/observability/__init__.py +0 -0
  40. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/observability/config.py +0 -0
  41. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/observability/models.py +0 -0
  42. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/observability/redactor.py +0 -0
  43. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/observability/session.py +0 -0
  44. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/performance/__init__.py +0 -0
  45. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/performance/instrumentation.py +0 -0
  46. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/performance/metrics.py +0 -0
  47. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/schema/__init__.py +0 -0
  48. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/schema/components.py +0 -0
  49. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/schema/connector.py +0 -0
  50. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/schema/operations.py +0 -0
  51. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/secrets.py +0 -0
  52. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/telemetry/__init__.py +0 -0
  53. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/telemetry/config.py +0 -0
  54. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/telemetry/events.py +0 -0
  55. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/telemetry/tracker.py +0 -0
  56. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/types.py +0 -0
  57. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/utils.py +0 -0
  58. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/validation.py +0 -0
  59. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/_vendored/connector_sdk/validation_replication.py +0 -0
  60. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/connector_model.py +0 -0
  61. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/models.py +16 -16
  62. {airbyte_agent_shopify-0.1.24 → airbyte_agent_shopify-0.1.26}/airbyte_agent_shopify/types.py +0 -0
@@ -41,6 +41,9 @@ In hosted mode, you first create a connector via the Airbyte API (providing your
41
41
  #### OAuth
42
42
  This authentication method isn't available for this connector.
43
43
 
44
+ #### Bring your own OAuth flow
45
+ This authentication method isn't available for this connector.
46
+
44
47
  #### Token
45
48
  Create a connector with Token credentials.
46
49
 
@@ -1,5 +1,15 @@
1
1
  # Shopify changelog
2
2
 
3
+ ## [0.1.26] - 2026-02-02
4
+ - Updated connector definition (YAML version 0.1.3)
5
+ - Source commit: 94024675
6
+ - SDK version: 0.1.0
7
+
8
+ ## [0.1.25] - 2026-02-02
9
+ - Updated connector definition (YAML version 0.1.3)
10
+ - Source commit: 9d9866b0
11
+ - SDK version: 0.1.0
12
+
3
13
  ## [0.1.24] - 2026-01-30
4
14
  - Updated connector definition (YAML version 0.1.3)
5
15
  - Source commit: b184da3e
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: airbyte-agent-shopify
3
- Version: 0.1.24
3
+ Version: 0.1.26
4
4
  Summary: Airbyte Shopify Connector for AI platforms
5
5
  Project-URL: Homepage, https://github.com/airbytehq/airbyte-agent-connectors
6
6
  Project-URL: Documentation, https://docs.airbyte.com/ai-agents/
@@ -171,7 +171,7 @@ See the official [Shopify API reference](https://shopify.dev/docs/api/admin-rest
171
171
 
172
172
  ## Version information
173
173
 
174
- - **Package version:** 0.1.24
174
+ - **Package version:** 0.1.26
175
175
  - **Connector version:** 0.1.3
176
- - **Generated with Connector SDK commit SHA:** b184da3e22ef8521d2eeebf3c96a0fe8da2424f5
176
+ - **Generated with Connector SDK commit SHA:** 940246757c7476ed4edd7d16b873ebe54ea2b456
177
177
  - **Changelog:** [View changelog](https://github.com/airbytehq/airbyte-agent-connectors/blob/main/connectors/shopify/CHANGELOG.md)
@@ -138,7 +138,7 @@ See the official [Shopify API reference](https://shopify.dev/docs/api/admin-rest
138
138
 
139
139
  ## Version information
140
140
 
141
- - **Package version:** 0.1.24
141
+ - **Package version:** 0.1.26
142
142
  - **Connector version:** 0.1.3
143
- - **Generated with Connector SDK commit SHA:** b184da3e22ef8521d2eeebf3c96a0fe8da2424f5
143
+ - **Generated with Connector SDK commit SHA:** 940246757c7476ed4edd7d16b873ebe54ea2b456
144
144
  - **Changelog:** [View changelog](https://github.com/airbytehq/airbyte-agent-connectors/blob/main/connectors/shopify/CHANGELOG.md)
@@ -19,8 +19,8 @@ from .models import (
19
19
  Refund,
20
20
  Order,
21
21
  OrderList,
22
- ProductVariant,
23
22
  ProductImage,
23
+ ProductVariant,
24
24
  Product,
25
25
  ProductList,
26
26
  ProductVariantList,
@@ -193,8 +193,8 @@ __all__ = [
193
193
  "Refund",
194
194
  "Order",
195
195
  "OrderList",
196
- "ProductVariant",
197
196
  "ProductImage",
197
+ "ProductVariant",
198
198
  "Product",
199
199
  "ProductList",
200
200
  "ProductVariantList",
@@ -161,24 +161,80 @@ class AirbyteCloudClient:
161
161
  connector_id = connectors[0]["id"]
162
162
  return connector_id
163
163
 
164
+ async def initiate_oauth(
165
+ self,
166
+ definition_id: str,
167
+ external_user_id: str,
168
+ redirect_url: str,
169
+ ) -> str:
170
+ """Initiate a server-side OAuth flow.
171
+
172
+ Starts the OAuth flow for a connector. Returns a consent URL where the
173
+ end user should be redirected to grant access. After completing consent,
174
+ they'll be redirected to your redirect_url with a `server_side_oauth_secret_id`
175
+ query parameter that can be used with `create_source()`.
176
+
177
+ Args:
178
+ definition_id: Connector definition UUID
179
+ external_user_id: Workspace identifier
180
+ redirect_url: URL where users will be redirected after OAuth consent
181
+
182
+ Returns:
183
+ The OAuth consent URL
184
+
185
+ Raises:
186
+ httpx.HTTPStatusError: If the request fails
187
+
188
+ Example:
189
+ consent_url = await client.initiate_oauth(
190
+ definition_id="d8313939-3782-41b0-be29-b3ca20d8dd3a",
191
+ external_user_id="my-workspace",
192
+ redirect_url="https://myapp.com/oauth/callback",
193
+ )
194
+ # Redirect user to: consent_url
195
+ # After consent: https://myapp.com/oauth/callback?server_side_oauth_secret_id=...
196
+ """
197
+ token = await self.get_bearer_token()
198
+ url = f"{self.API_BASE_URL}/api/v1/integrations/connectors/oauth/initiate"
199
+ headers = {"Authorization": f"Bearer {token}"}
200
+ request_body = {
201
+ "external_user_id": external_user_id,
202
+ "definition_id": definition_id,
203
+ "redirect_url": redirect_url,
204
+ }
205
+
206
+ response = await self._http_client.post(url, json=request_body, headers=headers)
207
+ response.raise_for_status()
208
+ return response.json()["consent_url"]
209
+
164
210
  async def create_source(
165
211
  self,
166
212
  name: str,
167
213
  connector_definition_id: str,
168
214
  external_user_id: str,
169
- credentials: dict[str, Any],
215
+ credentials: dict[str, Any] | None = None,
170
216
  replication_config: dict[str, Any] | None = None,
217
+ server_side_oauth_secret_id: str | None = None,
218
+ source_template_id: str | None = None,
171
219
  ) -> str:
172
- """Create a new source on Airbyte Agent Platform.
220
+ """Create a new source on Airbyte Cloud.
221
+
222
+ Supports two authentication modes:
223
+ 1. Direct credentials: Provide `credentials` dict
224
+ 2. Server-side OAuth: Provide `server_side_oauth_secret_id` from OAuth flow
173
225
 
174
226
  Args:
175
227
  name: Source name
176
228
  connector_definition_id: UUID of the connector definition
177
229
  external_user_id: User identifier
178
- credentials: Connector auth config dict
230
+ credentials: Connector auth config dict. Required unless using OAuth.
179
231
  replication_config: Optional replication settings (e.g., start_date for
180
232
  connectors with x-airbyte-replication-config). Required for REPLICATION
181
233
  mode sources like Intercom.
234
+ server_side_oauth_secret_id: OAuth secret ID from initiate_oauth redirect.
235
+ When provided, credentials are not required.
236
+ source_template_id: Source template ID. Required when organization has
237
+ multiple source templates for this connector type.
182
238
 
183
239
  Returns:
184
240
  The created source ID (UUID string)
@@ -187,19 +243,21 @@ class AirbyteCloudClient:
187
243
  httpx.HTTPStatusError: If creation fails
188
244
 
189
245
  Example:
246
+ # With direct credentials:
190
247
  source_id = await client.create_source(
191
- name="My Gong Source",
192
- connector_definition_id="32382e40-3b49-4b99-9c5c-4076501914e7",
248
+ name="My Intercom Source",
249
+ connector_definition_id="d8313939-3782-41b0-be29-b3ca20d8dd3a",
193
250
  external_user_id="my-workspace",
194
- credentials={"access_key": "...", "access_key_secret": "..."}
251
+ credentials={"access_token": "..."},
252
+ replication_config={"start_date": "2024-01-01T00:00:00Z"}
195
253
  )
196
254
 
197
- # For REPLICATION mode sources (e.g., Intercom):
255
+ # With server-side OAuth:
198
256
  source_id = await client.create_source(
199
257
  name="My Intercom Source",
200
258
  connector_definition_id="d8313939-3782-41b0-be29-b3ca20d8dd3a",
201
259
  external_user_id="my-workspace",
202
- credentials={"access_token": "..."},
260
+ server_side_oauth_secret_id="airbyte_oauth_..._secret_...",
203
261
  replication_config={"start_date": "2024-01-01T00:00:00Z"}
204
262
  )
205
263
  """
@@ -211,11 +269,16 @@ class AirbyteCloudClient:
211
269
  "name": name,
212
270
  "definition_id": connector_definition_id,
213
271
  "external_user_id": external_user_id,
214
- "credentials": credentials,
215
272
  }
216
273
 
274
+ if credentials is not None:
275
+ request_body["credentials"] = credentials
217
276
  if replication_config is not None:
218
277
  request_body["replication_config"] = replication_config
278
+ if server_side_oauth_secret_id is not None:
279
+ request_body["server_side_oauth_secret_id"] = server_side_oauth_secret_id
280
+ if source_template_id is not None:
281
+ request_body["source_template_id"] = source_template_id
219
282
 
220
283
  response = await self._http_client.post(url, json=request_body, headers=headers)
221
284
  response.raise_for_status()
@@ -10,7 +10,7 @@ from enum import StrEnum
10
10
  from typing import Any, Dict
11
11
  from uuid import UUID
12
12
 
13
- from pydantic import BaseModel, ConfigDict, Field, field_validator
13
+ from pydantic import BaseModel, ConfigDict, Field, field_validator, model_validator
14
14
  from pydantic_core import Url
15
15
 
16
16
  from .extensions import CacheConfig, ReplicationConfig, RetryConfig
@@ -210,3 +210,34 @@ class Server(BaseModel):
210
210
  raise ValueError("Server URL cannot be empty")
211
211
  # Allow both absolute URLs and relative paths
212
212
  return v
213
+
214
+ @model_validator(mode="after")
215
+ def validate_replication_environment_mapping(self) -> "Server":
216
+ """Validate that x-airbyte-replication-environment-mapping sources exist in variables.
217
+
218
+ For simple mappings like {"subdomain": "subdomain"}, the key is the source variable.
219
+ For transform mappings like {"domain": {"source": "subdomain", "format": "..."}},
220
+ the "source" field is the source variable.
221
+ """
222
+ env_mapping = self.x_airbyte_replication_environment_mapping
223
+ if not env_mapping or not self.variables:
224
+ return self
225
+
226
+ variable_names = set(self.variables.keys())
227
+
228
+ for env_key, mapping_value in env_mapping.items():
229
+ if isinstance(mapping_value, str):
230
+ source_var = env_key
231
+ elif isinstance(mapping_value, EnvironmentMappingTransform):
232
+ source_var = mapping_value.source
233
+ else:
234
+ continue
235
+
236
+ if source_var not in variable_names:
237
+ available = ", ".join(sorted(variable_names)) if variable_names else "(none)"
238
+ raise ValueError(
239
+ f"x-airbyte-replication-environment-mapping: source variable '{source_var}' "
240
+ f"not found in server variables. Available: {available}"
241
+ )
242
+
243
+ return self
@@ -14,7 +14,7 @@ are implemented.
14
14
 
15
15
  from typing import Literal
16
16
 
17
- from pydantic import BaseModel, ConfigDict, Field
17
+ from pydantic import BaseModel, ConfigDict, Field, model_validator
18
18
 
19
19
 
20
20
  class PaginationConfig(BaseModel):
@@ -252,6 +252,24 @@ class ReplicationConfig(BaseModel):
252
252
  description="Mapping from replication_config field names to source_config field names",
253
253
  )
254
254
 
255
+ @model_validator(mode="after")
256
+ def validate_replication_config_key_mapping(self) -> "ReplicationConfig":
257
+ """Validate that replication_config_key_mapping keys exist in properties.
258
+
259
+ The mapping is: {local_key: airbyte_path}
260
+ We validate that local_key exists in our properties.
261
+ """
262
+ if self.replication_config_key_mapping and self.properties:
263
+ property_names = set(self.properties.keys())
264
+ for local_key, airbyte_path in self.replication_config_key_mapping.items():
265
+ if local_key not in property_names:
266
+ available = ", ".join(sorted(property_names)) if property_names else "(none)"
267
+ raise ValueError(
268
+ f"replication_config_key_mapping: local key '{local_key}' "
269
+ f"(mapped to '{airbyte_path}') not found in properties. Available: {available}"
270
+ )
271
+ return self
272
+
255
273
 
256
274
  class CacheConfig(BaseModel):
257
275
  """
@@ -81,6 +81,21 @@ class AuthConfigOption(BaseModel):
81
81
  description="Mapping from source config paths (e.g., 'credentials.api_key') to auth config keys for direct connectors",
82
82
  )
83
83
 
84
+ @model_validator(mode="after")
85
+ def validate_replication_auth_key_mapping(self) -> "AuthConfigOption":
86
+ """Validate that replication_auth_key_mapping target keys exist in properties."""
87
+ if self.replication_auth_key_mapping and self.properties:
88
+ property_names = set(self.properties.keys())
89
+ for airbyte_path, our_key in self.replication_auth_key_mapping.items():
90
+ if our_key not in property_names:
91
+ option_context = f"oneOf option '{self.title}'" if self.title else "oneOf option"
92
+ available = ", ".join(sorted(property_names)) if property_names else "(none)"
93
+ raise ValueError(
94
+ f"replication_auth_key_mapping in {option_context}: target key '{our_key}' "
95
+ f"(mapped from '{airbyte_path}') not found in properties. Available: {available}"
96
+ )
97
+ return self
98
+
84
99
 
85
100
  class AirbyteAuthConfig(BaseModel):
86
101
  """
@@ -146,8 +161,30 @@ class AirbyteAuthConfig(BaseModel):
146
161
  if not self.auth_mapping:
147
162
  raise ValueError("Single auth option must have auth_mapping")
148
163
 
164
+ # Validate replication_auth_key_mapping targets exist in properties
165
+ if self.replication_auth_key_mapping and self.properties:
166
+ self._validate_replication_auth_key_mapping(self.replication_auth_key_mapping, self.properties, context="x-airbyte-auth-config")
167
+
149
168
  return self
150
169
 
170
+ @staticmethod
171
+ def _validate_replication_auth_key_mapping(mapping: Dict[str, str], properties: Dict[str, AuthConfigFieldSpec], context: str) -> None:
172
+ """Validate that replication_auth_key_mapping target keys exist in properties.
173
+
174
+ Args:
175
+ mapping: The replication_auth_key_mapping dict (airbyte_path -> our_key)
176
+ properties: The properties dict from x-airbyte-auth-config
177
+ context: Context string for error messages
178
+ """
179
+ property_names = set(properties.keys())
180
+ for airbyte_path, our_key in mapping.items():
181
+ if our_key not in property_names:
182
+ available = ", ".join(sorted(property_names)) if property_names else "(none)"
183
+ raise ValueError(
184
+ f"replication_auth_key_mapping in {context}: target key '{our_key}' "
185
+ f"(mapped from '{airbyte_path}') not found in properties. Available: {available}"
186
+ )
187
+
151
188
 
152
189
  class SecurityScheme(BaseModel):
153
190
  """
@@ -1088,6 +1088,7 @@ class ShopifyConnector:
1088
1088
  auth_config: "ShopifyAuthConfig",
1089
1089
  name: str | None = None,
1090
1090
  replication_config: dict[str, Any] | None = None,
1091
+ source_template_id: str | None = None,
1091
1092
  ) -> "ShopifyConnector":
1092
1093
  """
1093
1094
  Create a new hosted connector on Airbyte Cloud.
@@ -1104,6 +1105,8 @@ class ShopifyConnector:
1104
1105
  name: Optional source name (defaults to connector name + external_user_id)
1105
1106
  replication_config: Optional replication settings dict.
1106
1107
  Required for connectors with x-airbyte-replication-config (REPLICATION mode sources).
1108
+ source_template_id: Source template ID. Required when organization has
1109
+ multiple source templates for this connector type.
1107
1110
 
1108
1111
  Returns:
1109
1112
  A ShopifyConnector instance configured in hosted mode
@@ -1120,6 +1123,7 @@ class ShopifyConnector:
1120
1123
  # Use the connector
1121
1124
  result = await connector.execute("entity", "list", {})
1122
1125
  """
1126
+
1123
1127
  from ._vendored.connector_sdk.cloud_utils import AirbyteCloudClient
1124
1128
 
1125
1129
  client = AirbyteCloudClient(
@@ -1128,8 +1132,8 @@ class ShopifyConnector:
1128
1132
  )
1129
1133
 
1130
1134
  try:
1131
- # Build credentials from auth_config
1132
- credentials = auth_config.model_dump(exclude_none=True)
1135
+ # Build credentials from auth_config (if provided)
1136
+ credentials = auth_config.model_dump(exclude_none=True) if auth_config else None
1133
1137
  replication_config_dict = replication_config.model_dump(exclude_none=True) if replication_config else None
1134
1138
 
1135
1139
  # Create source on Airbyte Cloud
@@ -1140,6 +1144,7 @@ class ShopifyConnector:
1140
1144
  external_user_id=external_user_id,
1141
1145
  credentials=credentials,
1142
1146
  replication_config=replication_config_dict,
1147
+ source_template_id=source_template_id,
1143
1148
  )
1144
1149
  finally:
1145
1150
  await client.close()
@@ -1153,6 +1158,7 @@ class ShopifyConnector:
1153
1158
 
1154
1159
 
1155
1160
 
1161
+
1156
1162
  class CustomersQuery:
1157
1163
  """
1158
1164
  Query class for Customers entity operations.
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "airbyte-agent-shopify"
3
- version = "0.1.24"
3
+ version = "0.1.26"
4
4
  description = "Airbyte Shopify Connector for AI platforms"
5
5
  readme = "README.md"
6
6
  requires-python = ">=3.13"
@@ -329,6 +329,22 @@ class OrderList(BaseModel):
329
329
 
330
330
  orders: Union[list[Order], Any] = Field(default=None)
331
331
 
332
+ class ProductImage(BaseModel):
333
+ """A product image"""
334
+ model_config = ConfigDict(extra="allow", populate_by_name=True)
335
+
336
+ id: Union[int, Any] = Field(default=None)
337
+ product_id: Union[int | None, Any] = Field(default=None)
338
+ position: Union[int | None, Any] = Field(default=None)
339
+ created_at: Union[str | None, Any] = Field(default=None)
340
+ updated_at: Union[str | None, Any] = Field(default=None)
341
+ alt: Union[str | None, Any] = Field(default=None)
342
+ width: Union[int | None, Any] = Field(default=None)
343
+ height: Union[int | None, Any] = Field(default=None)
344
+ src: Union[str | None, Any] = Field(default=None)
345
+ variant_ids: Union[list[int] | None, Any] = Field(default=None)
346
+ admin_graphql_api_id: Union[str | None, Any] = Field(default=None)
347
+
332
348
  class ProductVariant(BaseModel):
333
349
  """A product variant"""
334
350
  model_config = ConfigDict(extra="allow", populate_by_name=True)
@@ -360,22 +376,6 @@ class ProductVariant(BaseModel):
360
376
  requires_shipping: Union[bool | None, Any] = Field(default=None)
361
377
  admin_graphql_api_id: Union[str | None, Any] = Field(default=None)
362
378
 
363
- class ProductImage(BaseModel):
364
- """A product image"""
365
- model_config = ConfigDict(extra="allow", populate_by_name=True)
366
-
367
- id: Union[int, Any] = Field(default=None)
368
- product_id: Union[int | None, Any] = Field(default=None)
369
- position: Union[int | None, Any] = Field(default=None)
370
- created_at: Union[str | None, Any] = Field(default=None)
371
- updated_at: Union[str | None, Any] = Field(default=None)
372
- alt: Union[str | None, Any] = Field(default=None)
373
- width: Union[int | None, Any] = Field(default=None)
374
- height: Union[int | None, Any] = Field(default=None)
375
- src: Union[str | None, Any] = Field(default=None)
376
- variant_ids: Union[list[int] | None, Any] = Field(default=None)
377
- admin_graphql_api_id: Union[str | None, Any] = Field(default=None)
378
-
379
379
  class Product(BaseModel):
380
380
  """A Shopify product"""
381
381
  model_config = ConfigDict(extra="allow", populate_by_name=True)