-
-
Notifications
You must be signed in to change notification settings - Fork 94
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Implement Polls #1922
base: master
Are you sure you want to change the base?
Implement Polls #1922
Conversation
Having an `attrs` class with `weakref_slot=False` at the same time as a predefined `__slots__` will cause `attrs` to throw a fit: TypeError: 'property' object is not iterable
...oops Signed-off-by: PythonTryHard <31789326+PythonTryHard@users.noreply.github.com>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here are my suggestions so far for the code, but it looks great otherwise.
def __init__( | ||
self, | ||
question: str, | ||
duration: int, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It would be nice to be able to set the duration via a datetime.timedelta
object at minimum, and maybe support a datetime.datetime
object, that strips of datetime.datetime.now()
? not sure about the second one, but definitely the first.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Okay, I can't seem to push anything to GH, so I will just send it here.
def _serialize_poll_partial_emoji(self, emoji: typing.Optional[emoji_models.Emoji]) -> data_binding.JSONObject:
if isinstance(emoji, emoji_models.UnicodeEmoji):
return {"name": emoji.name}
elif isinstance(emoji, emoji_models.CustomEmoji):
return {"name": emoji.name, "id": emoji.name}
return {}
def _serialize_poll_media(self, poll_media: poll_models.PollMedia) -> data_binding.JSONObject:
# FIXME: Typing is **very** dodgy here. Revise this before shipping.
serialised_poll_media: typing.MutableMapping[str, typing.Any] = {"text": poll_media.text}
answer_emoji = self._serialize_poll_partial_emoji(poll_media.emoji)
if answer_emoji:
serialised_poll_media["emoji"] = answer_emoji
return serialised_poll_media
def serialize_poll(self, poll: poll_models.PollCreate) -> data_binding.JSONObject:
answers: typing.MutableSequence[typing.Any] = []
for answer_id, answer in poll.answers.items():
answers.append({"answer_id": answer_id, "poll_media": self._serialize_poll_media(answer.poll_media)})
return {
"question": self._serialize_poll_media(poll.question),
"answers": answers,
"expiry": poll.duration,
"allow_multiple_options": poll.allow_multiselect,
"layout_type": poll.layout_type,
} |
added `_serialize_poll_media()` and fixed polls question serialisation.
The removal of the counter has been added, so that the user can choose there own options, however, they may need to add a check because in its current form, you can override other answers.
Co-authored-by: MPlatypus <contact@mplaty.com>
"Apparently".
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Overall really nice pr, thanks!
class PartialPoll: | ||
"""Base class for all poll objects.""" | ||
|
||
__slots__: typing.Sequence[str] = ("_question", "_answers", "_allow_multiselect", "_layout_type", "_counter") | ||
|
||
def __init__(self, question: str, allow_multiselect: bool, layout_type: typing.Union[int, PollLayoutType]): | ||
self._question = PollMedia(text=question) # Only text is supported for question | ||
self._allow_multiselect = allow_multiselect | ||
self._layout_type = layout_type |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This object is a bit too confusing. It should be more like embed (or it should be moved as a special endpoint class)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Considering it's a certified Discord moment™️ by having a separate poll object and poll creation object, I have absolutely no idea how to implement the two classes elegantly without copy-pasting code. Hence the hacky PartialPoll
.
Is it against stuffs like PartialMessage
, PartialGuild
, etc.? Yes. Do I have any other ideas? No, sadly.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also not happy with it at the current time
- De-underscore `answer_payload`. - Remove unnecessary `dict.get` from `deserialize_poll`. - Streamline serialization of poll in `_build_message_payload`. - API endpoint naming consistency. - Correct typing targeting Python 3.8 Co-authored-by: davfsa <davfsa@gmail.com> Signed-off-by: PythonTryHard <31789326+PythonTryHard@users.noreply.github.com>
Co-authored-by: MPlatypus <contact@mplaty.com>
Most, or all tests have been completed. - test_serialize_poll - test_deserialize_poll_vote_create_event - test_deserialize_poll_vote_delete_event
Co-authored by: MPlatypus <contact@mplaty.com>
Continuation of 51fe4e4
Running `nox -s pytest` would randomly throws an error for failing `assert isinstance(response, list)`. Failed on my machine, does not fail on Platy's machine. This fixes the issue by making sure `AsyncMock` returns a proper list instead of an `AsyncMock` object. Co-authored by: davfsa <davfsa@gmail.com> Thanks dav!
on_message_poll_vote_add on_message_poll_vote_remove fixed get_poll_answer test.
CI is green. Let's end this. |
@davfsa One final thing is left, waiting for your input |
|
||
answers: typing.MutableSequence[poll_models.PollAnswer] = [] | ||
for answer_payload in payload["answers"]: | ||
answer_id = answer_payload["answer_id"] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This could be missing in some cases according to the documentation:
https://discord.com/developers/docs/resources/poll#poll-answer-object-poll-answer-object-structure
Only sent as part of responses from Discord's API/Gateway.
GUILD_MESSAGE_POLLS = 1 << 24 | ||
"""Subscribes to the events listed below. | ||
|
||
* `MESSAGE_POLL_VOTE_ADD` | ||
* `MESSAGE_POLL_VOTE_REMOVE` | ||
""" | ||
|
||
DIRECT_MESSAGE_POLLS = 1 << 25 | ||
"""Subscribes to the events listed below. | ||
|
||
* `MESSAGE_POLL_VOTE_ADD` | ||
* `MESSAGE_POLL_VOTE_REMOVE` | ||
""" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Currently the docstring is the same, would be nice to differentiate between guild and DM in the docstring too
@@ -345,6 +345,10 @@ def compile_to_file( | |||
PATCH_STAGE_INSTANCE: typing.Final[Route] = Route(PATCH, "/stage-instances/{channel}") | |||
DELETE_STAGE_INSTANCE: typing.Final[Route] = Route(DELETE, "/stage-instances/{channel}") | |||
|
|||
# Polls | |||
GET_POLL_ANSWER: typing.Final[Route] = Route(GET, "/channels/{channel}/polls/{message}/answer/{answer}") | |||
POST_END_POLL: typing.Final[Route] = Route(POST, "/channels/{channel}/polls/{message}/expire") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I hate discord for the naming, but maybe
POST_END_POLL: typing.Final[Route] = Route(POST, "/channels/{channel}/polls/{message}/expire") | |
POST_EXPIRE_POLL: typing.Final[Route] = Route(POST, "/channels/{channel}/polls/{message}/expire") |
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
# SOFTWARE. | ||
"""Polls and poll-related objects.""" # TODO: Improve this docstring |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Todo indeed :)
class PartialPoll: | ||
"""Base class for all poll objects.""" | ||
|
||
__slots__: typing.Sequence[str] = ("_question", "_answers", "_allow_multiselect", "_layout_type", "_counter") | ||
|
||
def __init__(self, question: str, allow_multiselect: bool, layout_type: typing.Union[int, PollLayoutType]): | ||
self._question = PollMedia(text=question) # Only text is supported for question | ||
self._allow_multiselect = allow_multiselect | ||
self._layout_type = layout_type |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also not happy with it at the current time
return self | ||
|
||
|
||
class Poll(PartialPoll): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Overall not happy with the the layout of these objects, will add more meaningful comments when I play around with the design a bit.
I like the builder idea, but it feels a bit too over the place. Could be condensed way better
Summary
Implementing built-in Discord polls.
Checklist
nox
and all the pipelines have passed.Related issues
Resolves #1886