add campaign api (#274)
This commit is contained in:
@@ -2,6 +2,7 @@
|
||||
|
||||
from .usesend import UseSend, UseSendHTTPError
|
||||
from .domains import Domains # type: ignore
|
||||
from .campaigns import Campaigns # type: ignore
|
||||
from . import types
|
||||
|
||||
__all__ = ["UseSend", "UseSendHTTPError", "types", "Domains"]
|
||||
__all__ = ["UseSend", "UseSendHTTPError", "types", "Domains", "Campaigns"]
|
||||
|
||||
@@ -0,0 +1,68 @@
|
||||
"""Campaign resource client using TypedDict shapes (no Pydantic)."""
|
||||
from __future__ import annotations
|
||||
|
||||
from typing import Any, Dict, Optional, Tuple
|
||||
|
||||
from .types import (
|
||||
APIError,
|
||||
Campaign,
|
||||
CampaignCreate,
|
||||
CampaignCreateResponse,
|
||||
CampaignSchedule,
|
||||
CampaignScheduleResponse,
|
||||
CampaignActionResponse,
|
||||
)
|
||||
|
||||
|
||||
class Campaigns:
|
||||
"""Client for `/campaigns` endpoints."""
|
||||
|
||||
def __init__(self, usesend: "UseSend") -> None:
|
||||
self.usesend = usesend
|
||||
|
||||
def create(
|
||||
self, payload: CampaignCreate
|
||||
) -> Tuple[Optional[CampaignCreateResponse], Optional[APIError]]:
|
||||
data, err = self.usesend.post(
|
||||
"/campaigns",
|
||||
payload,
|
||||
)
|
||||
return (data, err) # type: ignore[return-value]
|
||||
|
||||
def get(
|
||||
self, campaign_id: str
|
||||
) -> Tuple[Optional[Campaign], Optional[APIError]]:
|
||||
data, err = self.usesend.get(
|
||||
f"/campaigns/{campaign_id}"
|
||||
)
|
||||
return (data, err) # type: ignore[return-value]
|
||||
|
||||
def schedule(
|
||||
self, campaign_id: str, payload: CampaignSchedule
|
||||
) -> Tuple[Optional[CampaignScheduleResponse], Optional[APIError]]:
|
||||
data, err = self.usesend.post(
|
||||
f"/campaigns/{campaign_id}/schedule",
|
||||
payload,
|
||||
)
|
||||
return (data, err) # type: ignore[return-value]
|
||||
|
||||
def pause(
|
||||
self, campaign_id: str
|
||||
) -> Tuple[Optional[CampaignActionResponse], Optional[APIError]]:
|
||||
data, err = self.usesend.post(
|
||||
f"/campaigns/{campaign_id}/pause",
|
||||
{},
|
||||
)
|
||||
return (data, err) # type: ignore[return-value]
|
||||
|
||||
def resume(
|
||||
self, campaign_id: str
|
||||
) -> Tuple[Optional[CampaignActionResponse], Optional[APIError]]:
|
||||
data, err = self.usesend.post(
|
||||
f"/campaigns/{campaign_id}/resume",
|
||||
{},
|
||||
)
|
||||
return (data, err) # type: ignore[return-value]
|
||||
|
||||
|
||||
from .usesend import UseSend # noqa: E402 pylint: disable=wrong-import-position
|
||||
@@ -15,14 +15,14 @@ from typing_extensions import NotRequired, Required, Literal
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
DomainStatus = Literal[
|
||||
'NOT_STARTED',
|
||||
'PENDING',
|
||||
'SUCCESS',
|
||||
'FAILED',
|
||||
'TEMPORARY_FAILURE',
|
||||
"NOT_STARTED",
|
||||
"PENDING",
|
||||
"SUCCESS",
|
||||
"FAILED",
|
||||
"TEMPORARY_FAILURE",
|
||||
]
|
||||
|
||||
DNSRecordType = Literal['MX', 'TXT']
|
||||
DNSRecordType = Literal["MX", "TXT"]
|
||||
|
||||
|
||||
class DNSRecord(TypedDict, total=False):
|
||||
@@ -99,24 +99,25 @@ class DomainDeleteResponse(TypedDict):
|
||||
success: bool
|
||||
message: str
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Emails
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
EmailEventStatus = Literal[
|
||||
'SCHEDULED',
|
||||
'QUEUED',
|
||||
'SENT',
|
||||
'DELIVERY_DELAYED',
|
||||
'BOUNCED',
|
||||
'REJECTED',
|
||||
'RENDERING_FAILURE',
|
||||
'DELIVERED',
|
||||
'OPENED',
|
||||
'CLICKED',
|
||||
'COMPLAINED',
|
||||
'FAILED',
|
||||
'CANCELLED',
|
||||
"SCHEDULED",
|
||||
"QUEUED",
|
||||
"SENT",
|
||||
"DELIVERY_DELAYED",
|
||||
"BOUNCED",
|
||||
"REJECTED",
|
||||
"RENDERING_FAILURE",
|
||||
"DELIVERED",
|
||||
"OPENED",
|
||||
"CLICKED",
|
||||
"COMPLAINED",
|
||||
"FAILED",
|
||||
"CANCELLED",
|
||||
]
|
||||
|
||||
|
||||
@@ -128,22 +129,22 @@ class EmailEvent(TypedDict, total=False):
|
||||
|
||||
|
||||
Email = TypedDict(
|
||||
'Email',
|
||||
"Email",
|
||||
{
|
||||
'id': str,
|
||||
'teamId': float,
|
||||
'to': Union[str, List[str]],
|
||||
'replyTo': NotRequired[Union[str, List[str]]],
|
||||
'cc': NotRequired[Union[str, List[str]]],
|
||||
'bcc': NotRequired[Union[str, List[str]]],
|
||||
'from': str,
|
||||
'subject': str,
|
||||
'html': str,
|
||||
'text': str,
|
||||
'createdAt': str,
|
||||
'updatedAt': str,
|
||||
'emailEvents': List[EmailEvent],
|
||||
}
|
||||
"id": str,
|
||||
"teamId": float,
|
||||
"to": Union[str, List[str]],
|
||||
"replyTo": NotRequired[Union[str, List[str]]],
|
||||
"cc": NotRequired[Union[str, List[str]]],
|
||||
"bcc": NotRequired[Union[str, List[str]]],
|
||||
"from": str,
|
||||
"subject": str,
|
||||
"html": str,
|
||||
"text": str,
|
||||
"createdAt": str,
|
||||
"updatedAt": str,
|
||||
"emailEvents": List[EmailEvent],
|
||||
},
|
||||
)
|
||||
|
||||
|
||||
@@ -157,40 +158,40 @@ class EmailUpdateResponse(TypedDict, total=False):
|
||||
|
||||
|
||||
EmailLatestStatus = Literal[
|
||||
'SCHEDULED',
|
||||
'QUEUED',
|
||||
'SENT',
|
||||
'DELIVERY_DELAYED',
|
||||
'BOUNCED',
|
||||
'REJECTED',
|
||||
'RENDERING_FAILURE',
|
||||
'DELIVERED',
|
||||
'OPENED',
|
||||
'CLICKED',
|
||||
'COMPLAINED',
|
||||
'FAILED',
|
||||
'CANCELLED',
|
||||
"SCHEDULED",
|
||||
"QUEUED",
|
||||
"SENT",
|
||||
"DELIVERY_DELAYED",
|
||||
"BOUNCED",
|
||||
"REJECTED",
|
||||
"RENDERING_FAILURE",
|
||||
"DELIVERED",
|
||||
"OPENED",
|
||||
"CLICKED",
|
||||
"COMPLAINED",
|
||||
"FAILED",
|
||||
"CANCELLED",
|
||||
]
|
||||
|
||||
|
||||
EmailListItem = TypedDict(
|
||||
'EmailListItem',
|
||||
"EmailListItem",
|
||||
{
|
||||
'id': str,
|
||||
'to': Union[str, List[str]],
|
||||
'replyTo': NotRequired[Union[str, List[str]]],
|
||||
'cc': NotRequired[Union[str, List[str]]],
|
||||
'bcc': NotRequired[Union[str, List[str]]],
|
||||
'from': str,
|
||||
'subject': str,
|
||||
'html': str,
|
||||
'text': str,
|
||||
'createdAt': str,
|
||||
'updatedAt': str,
|
||||
'latestStatus': EmailLatestStatus,
|
||||
'scheduledAt': str,
|
||||
'domainId': float,
|
||||
}
|
||||
"id": str,
|
||||
"to": Union[str, List[str]],
|
||||
"replyTo": NotRequired[Union[str, List[str]]],
|
||||
"cc": NotRequired[Union[str, List[str]]],
|
||||
"bcc": NotRequired[Union[str, List[str]]],
|
||||
"from": str,
|
||||
"subject": str,
|
||||
"html": str,
|
||||
"text": str,
|
||||
"createdAt": str,
|
||||
"updatedAt": str,
|
||||
"latestStatus": EmailLatestStatus,
|
||||
"scheduledAt": str,
|
||||
"domainId": float,
|
||||
},
|
||||
)
|
||||
|
||||
|
||||
@@ -205,23 +206,23 @@ class Attachment(TypedDict):
|
||||
|
||||
|
||||
EmailCreate = TypedDict(
|
||||
'EmailCreate',
|
||||
"EmailCreate",
|
||||
{
|
||||
'to': Required[Union[str, List[str]]],
|
||||
'from': Required[str],
|
||||
'subject': NotRequired[str],
|
||||
'templateId': NotRequired[str],
|
||||
'variables': NotRequired[Dict[str, str]],
|
||||
'replyTo': NotRequired[Union[str, List[str]]],
|
||||
'cc': NotRequired[Union[str, List[str]]],
|
||||
'bcc': NotRequired[Union[str, List[str]]],
|
||||
'text': NotRequired[str],
|
||||
'html': NotRequired[str],
|
||||
'attachments': NotRequired[List[Attachment]],
|
||||
'scheduledAt': NotRequired[Union[datetime, str]],
|
||||
'inReplyToId': NotRequired[str],
|
||||
'headers': NotRequired[Dict[str, str]],
|
||||
}
|
||||
"to": Required[Union[str, List[str]]],
|
||||
"from": Required[str],
|
||||
"subject": NotRequired[str],
|
||||
"templateId": NotRequired[str],
|
||||
"variables": NotRequired[Dict[str, str]],
|
||||
"replyTo": NotRequired[Union[str, List[str]]],
|
||||
"cc": NotRequired[Union[str, List[str]]],
|
||||
"bcc": NotRequired[Union[str, List[str]]],
|
||||
"text": NotRequired[str],
|
||||
"html": NotRequired[str],
|
||||
"attachments": NotRequired[List[Attachment]],
|
||||
"scheduledAt": NotRequired[Union[datetime, str]],
|
||||
"inReplyToId": NotRequired[str],
|
||||
"headers": NotRequired[Dict[str, str]],
|
||||
},
|
||||
)
|
||||
|
||||
|
||||
@@ -230,23 +231,23 @@ class EmailCreateResponse(TypedDict, total=False):
|
||||
|
||||
|
||||
EmailBatchItem = TypedDict(
|
||||
'EmailBatchItem',
|
||||
"EmailBatchItem",
|
||||
{
|
||||
'to': Required[Union[str, List[str]]],
|
||||
'from': Required[str],
|
||||
'subject': NotRequired[str],
|
||||
'templateId': NotRequired[str],
|
||||
'variables': NotRequired[Dict[str, str]],
|
||||
'replyTo': NotRequired[Union[str, List[str]]],
|
||||
'cc': NotRequired[Union[str, List[str]]],
|
||||
'bcc': NotRequired[Union[str, List[str]]],
|
||||
'text': NotRequired[str],
|
||||
'html': NotRequired[str],
|
||||
'attachments': NotRequired[List[Attachment]],
|
||||
'scheduledAt': NotRequired[Union[datetime, str]],
|
||||
'inReplyToId': NotRequired[str],
|
||||
'headers': NotRequired[Dict[str, str]],
|
||||
}
|
||||
"to": Required[Union[str, List[str]]],
|
||||
"from": Required[str],
|
||||
"subject": NotRequired[str],
|
||||
"templateId": NotRequired[str],
|
||||
"variables": NotRequired[Dict[str, str]],
|
||||
"replyTo": NotRequired[Union[str, List[str]]],
|
||||
"cc": NotRequired[Union[str, List[str]]],
|
||||
"bcc": NotRequired[Union[str, List[str]]],
|
||||
"text": NotRequired[str],
|
||||
"html": NotRequired[str],
|
||||
"attachments": NotRequired[List[Attachment]],
|
||||
"scheduledAt": NotRequired[Union[datetime, str]],
|
||||
"inReplyToId": NotRequired[str],
|
||||
"headers": NotRequired[Dict[str, str]],
|
||||
},
|
||||
)
|
||||
|
||||
|
||||
@@ -269,6 +270,7 @@ class EmailCancelResponse(TypedDict, total=False):
|
||||
# Contacts
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
|
||||
class ContactCreate(TypedDict, total=False):
|
||||
email: str
|
||||
firstName: Optional[str]
|
||||
@@ -335,11 +337,115 @@ class ContactDeleteResponse(TypedDict):
|
||||
success: bool
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Campaigns
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
Campaign = TypedDict(
|
||||
"Campaign",
|
||||
{
|
||||
"id": str,
|
||||
"name": str,
|
||||
"from": str,
|
||||
"subject": str,
|
||||
"previewText": Optional[str],
|
||||
"contactBookId": Optional[str],
|
||||
"html": Optional[str],
|
||||
"content": Optional[str],
|
||||
"status": str,
|
||||
"scheduledAt": Optional[str],
|
||||
"batchSize": int,
|
||||
"batchWindowMinutes": int,
|
||||
"total": int,
|
||||
"sent": int,
|
||||
"delivered": int,
|
||||
"opened": int,
|
||||
"clicked": int,
|
||||
"unsubscribed": int,
|
||||
"bounced": int,
|
||||
"hardBounced": int,
|
||||
"complained": int,
|
||||
"replyTo": List[str],
|
||||
"cc": List[str],
|
||||
"bcc": List[str],
|
||||
"createdAt": str,
|
||||
"updatedAt": str,
|
||||
},
|
||||
)
|
||||
|
||||
|
||||
CampaignCreate = TypedDict(
|
||||
"CampaignCreate",
|
||||
{
|
||||
"name": Required[str],
|
||||
"from": Required[str],
|
||||
"subject": Required[str],
|
||||
"previewText": NotRequired[str],
|
||||
"contactBookId": Required[str],
|
||||
"content": NotRequired[str],
|
||||
"html": NotRequired[str],
|
||||
"replyTo": NotRequired[Union[str, List[str]]],
|
||||
"cc": NotRequired[Union[str, List[str]]],
|
||||
"bcc": NotRequired[Union[str, List[str]]],
|
||||
"sendNow": NotRequired[bool],
|
||||
"scheduledAt": NotRequired[str],
|
||||
"batchSize": NotRequired[int],
|
||||
},
|
||||
)
|
||||
|
||||
|
||||
CampaignCreateResponse = TypedDict(
|
||||
"CampaignCreateResponse",
|
||||
{
|
||||
"id": str,
|
||||
"name": str,
|
||||
"from": str,
|
||||
"subject": str,
|
||||
"previewText": Optional[str],
|
||||
"contactBookId": Optional[str],
|
||||
"html": Optional[str],
|
||||
"content": Optional[str],
|
||||
"status": str,
|
||||
"scheduledAt": Optional[str],
|
||||
"batchSize": int,
|
||||
"batchWindowMinutes": int,
|
||||
"total": int,
|
||||
"sent": int,
|
||||
"delivered": int,
|
||||
"opened": int,
|
||||
"clicked": int,
|
||||
"unsubscribed": int,
|
||||
"bounced": int,
|
||||
"hardBounced": int,
|
||||
"complained": int,
|
||||
"replyTo": List[str],
|
||||
"cc": List[str],
|
||||
"bcc": List[str],
|
||||
"createdAt": str,
|
||||
"updatedAt": str,
|
||||
},
|
||||
)
|
||||
|
||||
|
||||
class CampaignSchedule(TypedDict, total=False):
|
||||
scheduledAt: Optional[str]
|
||||
batchSize: Optional[int]
|
||||
sendNow: Optional[bool]
|
||||
|
||||
|
||||
class CampaignScheduleResponse(TypedDict, total=False):
|
||||
success: bool
|
||||
|
||||
|
||||
class CampaignActionResponse(TypedDict, total=False):
|
||||
success: bool
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Common
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
|
||||
class APIError(TypedDict):
|
||||
code: str
|
||||
message: str
|
||||
|
||||
|
||||
@@ -72,6 +72,7 @@ class UseSend:
|
||||
self.emails = Emails(self)
|
||||
self.contacts = Contacts(self)
|
||||
self.domains = Domains(self)
|
||||
self.campaigns = Campaigns(self)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Internal request helper
|
||||
@@ -125,3 +126,4 @@ class UseSend:
|
||||
from .emails import Emails # noqa: E402 pylint: disable=wrong-import-position
|
||||
from .contacts import Contacts # noqa: E402 pylint: disable=wrong-import-position
|
||||
from .domains import Domains # type: ignore # noqa: E402
|
||||
from .campaigns import Campaigns # type: ignore # noqa: E402
|
||||
|
||||
Reference in New Issue
Block a user