shop management features
This commit is contained in:
@@ -55,6 +55,75 @@ class LoginResponse(BaseModel):
|
||||
user: UserResponse
|
||||
|
||||
|
||||
# NEW: Shop models
|
||||
class ShopCreate(BaseModel):
|
||||
shop_code: str = Field(..., min_length=3, max_length=50, description="Unique shop code (e.g., TECHSTORE)")
|
||||
shop_name: str = Field(..., min_length=1, max_length=200, description="Display name of the shop")
|
||||
description: Optional[str] = Field(None, max_length=2000, description="Shop description")
|
||||
contact_email: Optional[str] = None
|
||||
contact_phone: Optional[str] = None
|
||||
website: Optional[str] = None
|
||||
business_address: Optional[str] = None
|
||||
tax_number: Optional[str] = None
|
||||
|
||||
@field_validator('shop_code')
|
||||
def validate_shop_code(cls, v):
|
||||
# Convert to uppercase and check format
|
||||
v = v.upper().strip()
|
||||
if not v.replace('_', '').replace('-', '').isalnum():
|
||||
raise ValueError('Shop code must be alphanumeric (underscores and hyphens allowed)')
|
||||
return v
|
||||
|
||||
@field_validator('contact_email')
|
||||
def validate_contact_email(cls, v):
|
||||
if v and ('@' not in v or '.' not in v):
|
||||
raise ValueError('Invalid email format')
|
||||
return v.lower() if v else v
|
||||
|
||||
|
||||
class ShopUpdate(BaseModel):
|
||||
shop_name: Optional[str] = Field(None, min_length=1, max_length=200)
|
||||
description: Optional[str] = Field(None, max_length=2000)
|
||||
contact_email: Optional[str] = None
|
||||
contact_phone: Optional[str] = None
|
||||
website: Optional[str] = None
|
||||
business_address: Optional[str] = None
|
||||
tax_number: Optional[str] = None
|
||||
|
||||
@field_validator('contact_email')
|
||||
def validate_contact_email(cls, v):
|
||||
if v and ('@' not in v or '.' not in v):
|
||||
raise ValueError('Invalid email format')
|
||||
return v.lower() if v else v
|
||||
|
||||
|
||||
class ShopResponse(BaseModel):
|
||||
id: int
|
||||
shop_code: str
|
||||
shop_name: str
|
||||
description: Optional[str]
|
||||
owner_id: int
|
||||
contact_email: Optional[str]
|
||||
contact_phone: Optional[str]
|
||||
website: Optional[str]
|
||||
business_address: Optional[str]
|
||||
tax_number: Optional[str]
|
||||
is_active: bool
|
||||
is_verified: bool
|
||||
created_at: datetime
|
||||
updated_at: datetime
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
class ShopListResponse(BaseModel):
|
||||
shops: List[ShopResponse]
|
||||
total: int
|
||||
skip: int
|
||||
limit: int
|
||||
|
||||
|
||||
# Base Product Models with Marketplace Support
|
||||
class ProductBase(BaseModel):
|
||||
product_id: Optional[str] = None
|
||||
@@ -123,6 +192,41 @@ class ProductResponse(ProductBase):
|
||||
model_config = {"from_attributes": True}
|
||||
|
||||
|
||||
# NEW: Shop Product models
|
||||
class ShopProductCreate(BaseModel):
|
||||
product_id: str = Field(..., description="Product ID to add to shop")
|
||||
shop_product_id: Optional[str] = Field(None, description="Shop's internal product ID")
|
||||
shop_price: Optional[float] = Field(None, ge=0, description="Shop-specific price override")
|
||||
shop_sale_price: Optional[float] = Field(None, ge=0, description="Shop-specific sale price")
|
||||
shop_currency: Optional[str] = Field(None, description="Shop-specific currency")
|
||||
shop_availability: Optional[str] = Field(None, description="Shop-specific availability")
|
||||
shop_condition: Optional[str] = Field(None, description="Shop-specific condition")
|
||||
is_featured: bool = Field(False, description="Featured product flag")
|
||||
min_quantity: int = Field(1, ge=1, description="Minimum order quantity")
|
||||
max_quantity: Optional[int] = Field(None, ge=1, description="Maximum order quantity")
|
||||
|
||||
|
||||
class ShopProductResponse(BaseModel):
|
||||
id: int
|
||||
shop_id: int
|
||||
product: ProductResponse
|
||||
shop_product_id: Optional[str]
|
||||
shop_price: Optional[float]
|
||||
shop_sale_price: Optional[float]
|
||||
shop_currency: Optional[str]
|
||||
shop_availability: Optional[str]
|
||||
shop_condition: Optional[str]
|
||||
is_featured: bool
|
||||
is_active: bool
|
||||
min_quantity: int
|
||||
max_quantity: Optional[int]
|
||||
created_at: datetime
|
||||
updated_at: datetime
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
# Stock Models
|
||||
class StockBase(BaseModel):
|
||||
gtin: str = Field(..., min_length=1, description="GTIN is required")
|
||||
@@ -168,7 +272,7 @@ class StockSummaryResponse(BaseModel):
|
||||
class MarketplaceImportRequest(BaseModel):
|
||||
url: str = Field(..., description="URL to CSV file from marketplace")
|
||||
marketplace: str = Field(default="Letzshop", description="Name of the marketplace (e.g., Letzshop, Amazon, eBay)")
|
||||
shop_name: str = Field(..., min_length=1, description="Name of the shop these products belong to")
|
||||
shop_code: str = Field(..., description="Shop code to associate products with")
|
||||
batch_size: Optional[int] = Field(1000, gt=0, le=10000, description="Batch size for processing")
|
||||
|
||||
@field_validator('url')
|
||||
@@ -188,18 +292,18 @@ class MarketplaceImportRequest(BaseModel):
|
||||
pass
|
||||
return v.strip()
|
||||
|
||||
@field_validator('shop_name')
|
||||
@field_validator('shop_code')
|
||||
@classmethod
|
||||
def validate_shop_name(cls, v):
|
||||
if not v or not v.strip():
|
||||
raise ValueError('Shop name cannot be empty')
|
||||
return v.strip()
|
||||
def validate_shop_code(cls, v):
|
||||
return v.upper().strip()
|
||||
|
||||
|
||||
class MarketplaceImportJobResponse(BaseModel):
|
||||
job_id: int
|
||||
status: str
|
||||
marketplace: str
|
||||
shop_id: int
|
||||
shop_code: Optional[str] = None # Will be populated from shop relationship
|
||||
shop_name: str
|
||||
message: Optional[str] = None
|
||||
imported: Optional[int] = 0
|
||||
|
||||
Reference in New Issue
Block a user