From: Christian Heller Date: Mon, 4 Aug 2025 20:17:41 +0000 (+0200) Subject: Perform SASL during CAP negotation, handle success case. X-Git-Url: https://plomlompom.com/repos/%7B%7B%20web_path%20%7D%7D/%7B%7Bprefix%7D%7D/%7B%7Bprefix%7D%7D/add_structured?a=commitdiff_plain;h=e17ed70f7345a66d309b62016f497eee3057b9eb;p=ircplom Perform SASL during CAP negotation, handle success case. --- diff --git a/ircplom/client.py b/ircplom/client.py index e9f367a..31fad38 100644 --- a/ircplom/client.py +++ b/ircplom/client.py @@ -79,9 +79,11 @@ class _CapsManager: self._send = sender self._challenges: dict[str, bool] = {} self._dict: dict[str, _ServerCapability] = {} + self.auth_wait = False def clear(self) -> None: 'Reset all negotiation knowledge to zero.' + self.auth_wait = False self._challenges.clear() self._dict.clear() @@ -91,7 +93,7 @@ class _CapsManager: self.clear() self.challenge('LS', '302') return [] - if self._challenged('END'): + if self._challenged('END') or self.auth_wait: return [f'ignoring post-END CAP message not NEW, DEL: {params}'] match params[0]: case 'LS' | 'LIST': @@ -101,7 +103,10 @@ class _CapsManager: self._challenge_set(f'REQ:{cap_name}', done=True) self._dict[cap_name].enabled = params[0] == 'ACK' if self._challenge_met('LIST'): - self.challenge('END') + if self.could_sasl_plain: + self.auth_wait = True + else: + self.challenge('END') return (['server capabilities (enabled: "+"):'] + [cap.str_for_log(cap_name) for cap_name, cap in self._dict.items()]) @@ -124,8 +129,7 @@ class _CapsManager: @property def could_sasl_plain(self) -> bool: 'Whether opportunity for some SASL AUTHENTICATE PLAIN attempt.' - return (self._challenged('END') - and 'sasl' in self._dict + return ('sasl' in self._dict and 'PLAIN' in self._dict['sasl'].data.split(',')) def _challenge_met(self, step: str) -> bool: @@ -268,10 +272,11 @@ class Client(ABC, ClientQueueMixin): case 'CAP': for to_log in self._caps.process_msg(msg.params[1:]): self.log.add(to_log) - if self._caps.could_sasl_plain and self.conn_setup.password: - # NB: spec recommends to AUTHENTICATE during, rather than - # after, CAPS negotation; opting for simplicity now instead - self.send(IrcMessage('AUTHENTICATE', ('PLAIN',))) + if self._caps.auth_wait and self._caps.could_sasl_plain: + if self.conn_setup.password: + self.send(IrcMessage('AUTHENTICATE', ('PLAIN',))) + else: + self._caps.challenge('END') case 'AUTHENTICATE': if msg.params == ('+',): auth = b64encode((self.conn_setup.nickname + '\0' + @@ -281,6 +286,8 @@ class Client(ABC, ClientQueueMixin): self.send(IrcMessage('AUTHENTICATE', (auth,))) case '904': self.log.alert('SASL authentication failed') + case '903' | '904': + self._caps.challenge('END') @dataclass