Releases: david-lev/pywa
2.7.0
What's Changed
Update with pip:
pip3 install -U pywa
- [flows] adding support for
ChipsSelector
component - [flows]
flow_name
instead offlow_id
while sending.flow_token
is now optional - [client] allowing to create flow with flow json and publish it with one request
- [utils] update flow json latest version to 6.3
chips_selector = FlowJSON(
version="6.3",
screens=[
Screen(
id="DEMO_SCREEN",
terminal=True,
title="Demo screen",
layout=Layout(
children=[
chips := ChipsSelector(
name="chips",
label="Personalize your experience",
description="Choose your interests to get personalized design ideas and solution",
max_selected_items=2,
data_source=[
DataSource(id="room_layout", title="🏡 Room layouts"),
DataSource(id="lighting", title="💡 Lighting"),
DataSource(id="renovation", title="🛠️ Renovation"),
DataSource(id="furnitures", title="📐 Room layouts"),
],
),
Footer(
label="Continue",
on_click_action=CompleteAction(
payload={
"chips": chips.ref,
}
),
),
],
),
),
],
)
from pywa import WhatsApp, types
wa = WhatsApp(...)
wa.create_flow(
name="my_flow",
categories=[types.FlowCategory.CUSTOMER_SUPPORT, types.FlowCategory.CONTACT_US],
endpoint_uri="https://my-server.com/my-flow",
flow_json=FlowJSON(...), # You can now set the json while creating the flow
publish=True # You can now publish the flow while creating it
)
wa.send_text(
to="...",
text="Let's get started!",
buttons=types.FlowButton(
title="Start",
flow_name="my_flow", # You can now use the flow name instead of the flow id
# flow_token="my_flow_token", # Flow token is now optional, default is `unused`
flow_action_screen="FIRST_SCREEN",
)
)
Full Changelog: 2.6.0...2.7.0
2.6.0
What's Changed
Update with pip:
pip3 install -U pywa
- [base_update] adding
shared_data
to help sharing context between handlers & filters (suggested by @mhaugestad in #94) - [flows] adding
FlowStr
- A helper class to create strings containing vars and math expressions without escaping and quoting them
from pywa import WhatsApp, types, filters
wa = WhatsApp(...)
@wa.on_message(filters.text, priority=100) # a higher priority will be executed first
def on_every_text_message(_: WhatsApp, m: types.Message):
m.shared_data["key"] = "some_value" # .shared_data is a dict that can be used to store data between handlers
m.continue_handling() # continue to the next handler
# access shared_data from filters
@wa.on_message(filters.new(lambda _, m: m.shared_data["key"] == "value"))
def handle_value(_: WhatsApp, m: types.Message):
if m.shared_data["other_key"] == "other_value":
...
from pywa.types.flows import *
FlowJSON(
screens=[
Screen(
id="START",
layout=Layout(
children=[
bill := TextInput(name="bill", input_type=InputType.NUMBER),
tip := TextInput(name="tip", input_type=InputType.NUMBER),
TextHeading(
# text=f"`'Your total bill is ' {bill.ref + (bill.ref * tip.ref / 100)}`",
text=FlowStr(
# avoid escaping and quoting vars and math expressions
string="Your total bill is {bill}",
bill=bill.ref + (bill.ref * tip.ref / 100),
)
),
]
),
)
]
)
Full Changelog: 2.5.2...2.6.0
2.5.2
What's Changed
Update with pip:
pip3 install -U pywa
- [handlers] sub-handlers ignoring flow requests with errors when
call_on_error
set to False - [handlers] fix
WhatsApp.on_flow_request
on class level to mark theFlowRequestHandler
instance - [callback] callback data generic can be
str
# my_handlers.py
from pywa import WhatsApp
from pywa.types import FlowRequest, FlowResponse
@WhatsApp.on_flow_request("/survey-flow")
def survey(_: WhatsApp, req: FlowRequest) -> FlowResponse:
raise NotImplementedError(req)
@survey.on_data_exchange(call_on_error=False) # The default
def on_good_rating(_: WhatsApp, req: FlowRequest) -> FlowResponse:
return req.respond(...)
@survey.on_data_exchange(call_on_error=True)
def on_error(_: WhatsApp, req: FlowRequest) -> None:
logging.error("Error in survey flow: %s", req.data)
# main.py
from . import my_handlers
# register the handlers from the module
wa = WhatsApp(..., handlers_modules=[my_handlers])
Full Changelog: 2.5.1...2.5.2
2.5.1
What's Changed
Update with pip:
pip3 install -U pywa
- [handlers] adding
on_init
,on_data_exchange
andon_back
decorators for flow request callback wrapper - [flows]
FlowRequest.respond
defaults to request flow token - [flows] adding
FlowRequest.token_no_longer_valid
shortcut - [flows] deprecate
FlowRequest.is_health_check
andFlowRequestActionType.PING
from pywa import WhatsApp, filters
from pywa.types import FlowRequest, FlowResponse
wa = WhatsApp(...)
@wa.on_flow_request("/survey-flow")
def survey(_: WhatsApp, req: FlowRequest) -> FlowResponse:
raise NotImplementedError(req)
@survey.on_init
def on_init(_: WhatsApp, req: FlowRequest) -> FlowResponse:
return req.respond(screen="SURVEY", data={"first_name": "David"})
@survey.on_data_exchange(screen="SURVEY", filters=filters.new(lambda _, r: int(r.data["rating"]) > 3))
def on_good_rating(_: WhatsApp, req: FlowRequest) -> FlowResponse:
return req.respond(screen="GOOD_RATING")
@survey.on_data_exchange(call_on_error=True)
def on_error(_: WhatsApp, req: FlowRequest) -> None:
logging.error("Error in survey flow: %s", req.data)
Full Changelog: 2.4.0...2.5.1
2.4.0
What's Changed
Update with pip:
pip3 install -U pywa
- [sent_message] adding
SentTemplate
withSentTemplateStatus
- [flows] adding
pattern
forTextInput
- [flows] adding support for
NavigationList
- [flows] defaulting action's payload to empty dict
- [flows] deprecating
ActionNext
andActionNextType
from pywa import WhatsApp
wa = WhatsApp(...)
sent_template_status = wa.send_template(...).status
from pywa.types.flows import *
FlowJSON(
version="6.2",
screens=[
Screen(
id="MENU",
title="Main Menu",
layout=Layout(
children=[
NavigationList(
name="menu",
list_items=[
NavigationItem(
id="1",
main_content=NavigationItemMainContent(
title="Contact Us",
description="Contact us now",
),
on_click_action=NavigateAction(next=Next(name="CONTACT")),
),
NavigationItem(
id="2",
main_content=NavigationItemMainContent(
title="About Us",
description="Learn more about us",
),
on_click_action=NavigateAction(next=Next(name="ABOUT")),
),
],
),
],
),
),
Screen(id="CONTACT", ...),
Screen(id="ABOUT", ...),
],
)
Full Changelog: 2.3.0...2.4.0
2.3.0
What's Changed
Update with pip:
pip3 install -U pywa
- [client] allowing to specify the callback url scope
- [client] expose methods to override callback url in waba and phone scopes
- [flows] typing
DataSource
to acceptRef
s
from pywa import WhatsApp, utils
import fastapi
fastapi_app = fastapi.FastAPI()
wa = WhatsApp(
phone_id=1234567890,
token="EAAEZC6hUxkTI...",
server=fastapi_app,
callback_url="https://example.com",
callback_url_scope=utils.CallbackURLScope.PHONE, # override phone callback url
verify_token="xyzxyz"
)
Full Changelog: 2.2.0...2.3.0
2.2.0
What's Changed
Update with pip:
pip3 install -U pywa
- [flows] adding
ScreenDataUpdate
to use inUpdateDataAction
- [flows] using math operators between math objs
- [flows] renaming
ActionNext
toNext
andActionNextType
toNextType
from pywa.types.flows import *
FlowJSON(
version="6.0",
screens=[
Screen(
id="DEMO_SCREEN",
data=[
is_txt_visible := ScreenData( # Declaring a data object
key="is_txt_visible",
example=False,
),
],
layout=Layout(
children=[
TextBody(
text="You checked the box!",
visible=is_txt_visible.ref, # Using the data
),
OptIn(
label="Show the text",
name="show_txt",
on_select_action=UpdateDataAction(
payload=[is_txt_visible.update(True)] # Updating the data
),
on_unselect_action=UpdateDataAction(
payload=[is_txt_visible.update(False)]
),
),
]
),
)
],
)
Full Changelog: 2.1.0...2.2.0
2.1.0
What's Changed
Update with pip:
pip3 install -U pywa
- [flows] adding
CalendarPicker
component - [flows] allow string concatenation with refs
- [flows] adding support for math expressions
- [flows] allowing to use condition in
visible
- [flows] new action: open url
- [flows] new action: update data
- [flows]
init_value
available outside form - [flows] allow to use
screen/ref
as shortcut for.ref_in(screen)
- [flows] separating
Action
and addingon_unselect_action
from pywa.types.flows import *
FlowJSON(
version="6.1",
screens=[
start := Screen(
id="START",
title="Start",
layout=Layout(
children=[
date_range := CalendarPicker( # new component
name="date_range",
label={
"start-date": "Select start date",
"end-date": "Select end date",
},
mode=CalendarPickerMode.RANGE,
),
..., # Footer with NavigateAction to `RESULT`
]
),
),
Screen(
id="RESULT",
title="Result",
layout=Layout(
children=[
name := TextInput(
name="name",
label="Enter your name",
input_type=InputType.TEXT,
),
birth_year := TextInput(
name="birth_year",
label="Enter birth year",
input_type=InputType.NUMBER,
init_value="1990", # `init_value` outside Form
),
TextBody(
# string concatenation with refs and math expressions
text=f"`'Hello, ' {name.ref} '. You are ' {2024 - birth_year.ref} ' years old.'`",
visible=birth_year.ref > 0, # condition in `visible`
),
OptIn(
name="rtd",
label="I read the docs",
on_click_action=OpenUrlAction( # Open URL action
url="https://pywa.readthedocs.io",
),
),
Footer(
label="Done",
on_click_action=CompleteAction( # `Action` is deprecated
payload={
"date_range": start/date_range.ref, # using screen/ref as shortcut
},
),
),
],
),
),
],
)
Full Changelog: 2.0.5...2.1.0
2.0.5
What's Changed
Update with pip:
pip3 install -U pywa
Note
This release is not compatible with 1.x versions. Please see the Migration Guide for instructions on updating from earlier versions.
- [client] fix
send_template
return typeSentMessage
Full Changelog: 2.0.4...2.0.5
2.0.4
What's Changed
Update with pip:
pip3 install -U pywa
Note
This release is not compatible with 1.x versions. Please see the Migration Guide for instructions on updating from earlier versions.
- [client] fix
reply_to_message
Full Changelog: 2.0.3...2.0.4