LCOV - code coverage report
Current view: top level - lib/encryption/utils - key_verification.dart (source / functions) Coverage Total Hit
Test: merged.info Lines: 87.0 % 660 574
Test Date: 2025-01-14 11:53:08 Functions: - 0 0

            Line data    Source code
       1              : /*
       2              :  *   Famedly Matrix SDK
       3              :  *   Copyright (C) 2020, 2021 Famedly GmbH
       4              :  *
       5              :  *   This program is free software: you can redistribute it and/or modify
       6              :  *   it under the terms of the GNU Affero General Public License as
       7              :  *   published by the Free Software Foundation, either version 3 of the
       8              :  *   License, or (at your option) any later version.
       9              :  *
      10              :  *   This program is distributed in the hope that it will be useful,
      11              :  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
      12              :  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
      13              :  *   GNU Affero General Public License for more details.
      14              :  *
      15              :  *   You should have received a copy of the GNU Affero General Public License
      16              :  *   along with this program.  If not, see <https://www.gnu.org/licenses/>.
      17              :  */
      18              : 
      19              : import 'dart:async';
      20              : import 'dart:convert';
      21              : import 'dart:typed_data';
      22              : 
      23              : import 'package:canonical_json/canonical_json.dart';
      24              : import 'package:olm/olm.dart' as olm;
      25              : import 'package:typed_data/typed_data.dart';
      26              : 
      27              : import 'package:matrix/encryption/encryption.dart';
      28              : import 'package:matrix/encryption/utils/base64_unpadded.dart';
      29              : import 'package:matrix/matrix.dart';
      30              : import 'package:matrix/src/utils/crypto/crypto.dart' as uc;
      31              : 
      32              : /*
      33              :     +-------------+                    +-----------+
      34              :     | AliceDevice |                    | BobDevice |
      35              :     +-------------+                    +-----------+
      36              :           |                                 |
      37              :           | (m.key.verification.request)    |
      38              :           |-------------------------------->| (ASK FOR VERIFICATION REQUEST)
      39              :           |                                 |
      40              :           |      (m.key.verification.ready) |
      41              :           |<--------------------------------|
      42              :           |                                 |
      43              :           |      (m.key.verification.start) | we will probably not send this
      44              :           |<--------------------------------| for simplicities sake
      45              :           |                                 |
      46              :           | m.key.verification.start        |
      47              :           |-------------------------------->| (ASK FOR VERIFICATION REQUEST)
      48              :           |                                 |
      49              :           |       m.key.verification.accept |
      50              :           |<--------------------------------|
      51              :           |                                 |
      52              :           | m.key.verification.key          |
      53              :           |-------------------------------->|
      54              :           |                                 |
      55              :           |          m.key.verification.key |
      56              :           |<--------------------------------|
      57              :           |                                 |
      58              :           |     COMPARE EMOJI / NUMBERS     |
      59              :           |                                 |
      60              :           | m.key.verification.mac          |
      61              :           |-------------------------------->|  success
      62              :           |                                 |
      63              :           |          m.key.verification.mac |
      64              :  success  |<--------------------------------|
      65              :           |                                 |
      66              : */
      67              : 
      68              : /// QR key verification
      69              : /// You create possible methods from `client.verificationMethods` on device A
      70              : /// and send a request using `request.start()` which calls `sendRequest()` your client
      71              : /// now is in `waitingAccept` state, where ideally your client would now show some
      72              : /// waiting indicator.
      73              : ///
      74              : /// On device B you now get a `m.key.verification.request`, you check the
      75              : /// `methods` from the request payload and see if anything is possible.
      76              : /// If not you cancel the request. (should this be cancelled? couldn't another device handle this?)
      77              : /// you the set the state to `askAccept`.
      78              : ///
      79              : /// Your client would now show a button to accept/decline the request.
      80              : /// The user can then use `acceptVerification()`to accept the verification which
      81              : /// then sends a `m.key.verification.ready`. This also calls `generateQrCode()`
      82              : /// in it which populates the `request.qrData` depending on the qr mode.
      83              : /// B now sets the state `askChoice`
      84              : ///
      85              : /// On device A you now get the ready event, which setups the `possibleMethods`
      86              : /// and `qrData` on A's side. Similarly A now sets their state to `askChoice`
      87              : ///
      88              : /// At his point both sides are on the `askChoice` state.
      89              : ///
      90              : /// BACKWARDS COMPATIBILITY HACK:
      91              : /// To work well with sdks prior to QR verification (0.20.5 and older), start will
      92              : /// be sent with ready itself if only sas is supported. This avoids weird glare
      93              : /// issues faced with start from both sides if clients are not on the same sdk
      94              : /// version (0.20.5 vs next)
      95              : /// https://matrix.to/#/!KBwfdofYJUmnsVoqwn:famedly.de/$wlHXlLQJdfrqKAF5KkuQrXydwOhY_uyqfH4ReasZqnA?via=neko.dev&via=famedly.de&via=lihotzki.de
      96              : 
      97              : /// Here your clients would ideally show a list of the `possibleMethods` and the
      98              : /// user can choose one. For QR specifically, you can show the QR code on the
      99              : /// device which supports showing the qr code and the device which supports
     100              : /// scanning can scan this code.
     101              : ///
     102              : /// Assuming device A scans device B's code, device A would now send a `m.key.verification.start`,
     103              : /// you do this using the `continueVerificatio()` method. You can pass
     104              : /// `m.reciprocate.v1` or `m.sas.v1` here, and also attach the qrData here.
     105              : /// This then calls `verifyQrData()` internally, which sets the `randomSharedSecretForQRCode`
     106              : /// to the one from the QR code. Device A is now set to `showQRSuccess` state and shows
     107              : /// a green sheild. (Maybe add a text below saying tell device B you scanned the
     108              : /// code successfully.)
     109              : ///
     110              : /// (some keys magic happens here, check `verifyQrData()`, `verifyKeysQR()` to know more)
     111              : ///
     112              : /// On device B you get the `m.key.verification.start` event. The secret sent in
     113              : /// the start request is then verified, device B is then set to the `confirmQRScan`
     114              : /// state. Your device should show a dialog to confirm from B that A's device shows
     115              : /// the green shield (is in the done state). Once B confirms this physically, you
     116              : /// call the `acceptQRScanConfirmation()` function, which then does some keys
     117              : /// magic and sets B's state to `done`.
     118              : ///
     119              : /// A gets the `m.key.verification.done` messsage and sends a done back, both
     120              : /// users can now dismiss the verification dialog safely.
     121              : 
     122              : enum KeyVerificationState {
     123              :   askChoice,
     124              :   askAccept,
     125              :   askSSSS,
     126              :   waitingAccept,
     127              :   askSas,
     128              :   showQRSuccess, // scanner after QR scan was successfull
     129              :   confirmQRScan, // shower after getting start
     130              :   waitingSas,
     131              :   done,
     132              :   error
     133              : }
     134              : 
     135              : enum KeyVerificationMethod { emoji, numbers, qrShow, qrScan, reciprocate }
     136              : 
     137            2 : bool isQrSupported(List knownVerificationMethods, List possibleMethods) {
     138            2 :   return knownVerificationMethods.contains(EventTypes.QRShow) &&
     139            2 :           possibleMethods.contains(EventTypes.QRScan) ||
     140            2 :       knownVerificationMethods.contains(EventTypes.QRScan) &&
     141            2 :           possibleMethods.contains(EventTypes.QRShow);
     142              : }
     143              : 
     144            1 : List<String> _intersect(List<String>? a, List<dynamic>? b) =>
     145            3 :     (b == null || a == null) ? [] : a.where(b.contains).toList();
     146              : 
     147            2 : List<String> _calculatePossibleMethods(
     148              :   List<String> knownMethods,
     149              :   List<dynamic> payloadMethods,
     150              : ) {
     151            2 :   final output = <String>[];
     152            2 :   final copyKnownMethods = List<String>.from(knownMethods);
     153            2 :   final copyPayloadMethods = List.from(payloadMethods);
     154              : 
     155              :   copyKnownMethods
     156            6 :       .removeWhere((element) => !copyPayloadMethods.contains(element));
     157              : 
     158              :   // remove qr modes for now, check if they are possible and add later
     159            6 :   copyKnownMethods.removeWhere((element) => element.startsWith('m.qr_code'));
     160            2 :   output.addAll(copyKnownMethods);
     161              : 
     162            2 :   if (isQrSupported(knownMethods, payloadMethods)) {
     163              :     // scan/show combo found, add whichever is known to us to our possible methods.
     164            2 :     if (payloadMethods.contains(EventTypes.QRScan) &&
     165            2 :         knownMethods.contains(EventTypes.QRShow)) {
     166            2 :       output.add(EventTypes.QRShow);
     167              :     }
     168            2 :     if (payloadMethods.contains(EventTypes.QRShow) &&
     169            2 :         knownMethods.contains(EventTypes.QRScan)) {
     170            2 :       output.add(EventTypes.QRScan);
     171              :     }
     172              :   } else {
     173            2 :     output.remove(EventTypes.Reciprocate);
     174              :   }
     175              : 
     176              :   return output;
     177              : }
     178              : 
     179            1 : List<int> _bytesToInt(Uint8List bytes, int totalBits) {
     180            1 :   final ret = <int>[];
     181              :   var current = 0;
     182              :   var numBits = 0;
     183            2 :   for (final byte in bytes) {
     184            2 :     for (final bit in [7, 6, 5, 4, 3, 2, 1, 0]) {
     185            1 :       numBits++;
     186            5 :       current |= ((byte >> bit) & 1) << (totalBits - numBits);
     187            1 :       if (numBits >= totalBits) {
     188            1 :         ret.add(current);
     189              :         current = 0;
     190              :         numBits = 0;
     191              :       }
     192              :     }
     193              :   }
     194              :   return ret;
     195              : }
     196              : 
     197            2 : _KeyVerificationMethod _makeVerificationMethod(
     198              :   String type,
     199              :   KeyVerification request,
     200              : ) {
     201            2 :   if (type == EventTypes.Sas) {
     202            2 :     return _KeyVerificationMethodSas(request: request);
     203              :   }
     204            2 :   if (type == EventTypes.Reciprocate) {
     205            2 :     return _KeyVerificationMethodQRReciprocate(request: request);
     206              :   }
     207            0 :   throw Exception('Unkown method type');
     208              : }
     209              : 
     210              : class KeyVerification {
     211              :   String? transactionId;
     212              :   final Encryption encryption;
     213            9 :   Client get client => encryption.client;
     214              :   final Room? room;
     215              :   final String userId;
     216              :   void Function()? onUpdate;
     217            6 :   String? get deviceId => _deviceId;
     218              :   String? _deviceId;
     219              :   bool startedVerification = false;
     220              :   _KeyVerificationMethod? _method;
     221              : 
     222              :   List<String> possibleMethods = [];
     223              :   List<String> oppositePossibleMethods = [];
     224              : 
     225              :   Map<String, dynamic>? startPayload;
     226              :   String? _nextAction;
     227              :   List<SignableKey> _verifiedDevices = [];
     228              : 
     229              :   DateTime lastActivity;
     230              :   String? lastStep;
     231              : 
     232              :   KeyVerificationState state = KeyVerificationState.waitingAccept;
     233              :   bool canceled = false;
     234              :   String? canceledCode;
     235              :   String? canceledReason;
     236            2 :   bool get isDone =>
     237            2 :       canceled ||
     238            8 :       {KeyVerificationState.error, KeyVerificationState.done}.contains(state);
     239              : 
     240              :   // qr stuff
     241              :   QRCode? qrCode;
     242              :   String? randomSharedSecretForQRCode;
     243              :   SignableKey? keyToVerify;
     244            3 :   KeyVerification({
     245              :     required this.encryption,
     246              :     this.room,
     247              :     required this.userId,
     248              :     String? deviceId,
     249              :     this.onUpdate,
     250              :   })  : _deviceId = deviceId,
     251            3 :         lastActivity = DateTime.now();
     252              : 
     253            3 :   void dispose() {
     254            6 :     Logs().i('[Key Verification] disposing object...');
     255            3 :     randomSharedSecretForQRCode = null;
     256            5 :     _method?.dispose();
     257              :   }
     258              : 
     259            2 :   static String? getTransactionId(Map<String, dynamic> payload) {
     260            2 :     return payload['transaction_id'] ??
     261            2 :         (payload['m.relates_to'] is Map
     262            2 :             ? payload['m.relates_to']['event_id']
     263              :             : null);
     264              :   }
     265              : 
     266            3 :   List<String> get knownVerificationMethods {
     267              :     final methods = <String>{};
     268            9 :     if (client.verificationMethods.contains(KeyVerificationMethod.numbers) ||
     269            3 :         client.verificationMethods.contains(KeyVerificationMethod.emoji)) {
     270            2 :       methods.add(EventTypes.Sas);
     271              :     }
     272              : 
     273              :     /// `qrCanWork` -  qr cannot work if we are verifying another master key but our own is unverified
     274           12 :     final qrCanWork = (userId == client.userID) ||
     275           14 :         ((client.userDeviceKeys[client.userID]?.masterKey?.verified ?? false));
     276              : 
     277            9 :     if (client.verificationMethods.contains(KeyVerificationMethod.qrShow) &&
     278              :         qrCanWork) {
     279            2 :       methods.add(EventTypes.QRShow);
     280            2 :       methods.add(EventTypes.Reciprocate);
     281              :     }
     282            9 :     if (client.verificationMethods.contains(KeyVerificationMethod.qrScan) &&
     283              :         qrCanWork) {
     284            2 :       methods.add(EventTypes.QRScan);
     285            2 :       methods.add(EventTypes.Reciprocate);
     286              :     }
     287              : 
     288            3 :     return methods.toList();
     289              :   }
     290              : 
     291              :   /// Once you get a ready event, i.e both sides are in a `askChoice` state,
     292              :   /// send either `m.reciprocate.v1` or `m.sas.v1` here. If you continue with
     293              :   /// qr, send the qrData you just scanned
     294            2 :   Future<void> continueVerification(
     295              :     String type, {
     296              :     Uint8List? qrDataRawBytes,
     297              :   }) async {
     298              :     bool qrChecksOut = false;
     299            4 :     if (possibleMethods.contains(type)) {
     300              :       if (qrDataRawBytes != null) {
     301            2 :         qrChecksOut = await verifyQrData(qrDataRawBytes);
     302              :         // after this scanners state is done
     303              :       }
     304            2 :       if (type != EventTypes.Reciprocate || qrChecksOut) {
     305            4 :         final method = _method = _makeVerificationMethod(type, this);
     306            2 :         await method.sendStart();
     307            2 :         if (type == EventTypes.Sas) {
     308            0 :           setState(KeyVerificationState.waitingAccept);
     309              :         }
     310            0 :       } else if (type == EventTypes.Reciprocate && !qrChecksOut) {
     311            0 :         Logs().e('[KeyVerification] qr did not check out');
     312            0 :         await cancel('m.invalid_key');
     313              :       }
     314              :     } else {
     315            4 :       Logs().e(
     316              :         '[KeyVerification] tried to continue verification with a unknown method',
     317              :       );
     318            2 :       await cancel('m.unknown_method');
     319              :     }
     320              :   }
     321              : 
     322            3 :   Future<void> sendRequest() async {
     323            3 :     await send(
     324              :       EventTypes.KeyVerificationRequest,
     325            3 :       {
     326            6 :         'methods': knownVerificationMethods,
     327            9 :         if (room == null) 'timestamp': DateTime.now().millisecondsSinceEpoch,
     328              :       },
     329              :     );
     330            3 :     startedVerification = true;
     331            3 :     setState(KeyVerificationState.waitingAccept);
     332            6 :     lastActivity = DateTime.now();
     333              :   }
     334              : 
     335            3 :   Future<void> start() async {
     336            3 :     if (room == null) {
     337            6 :       transactionId = client.generateUniqueTransactionId();
     338              :     }
     339            9 :     if (encryption.crossSigning.enabled &&
     340            9 :         !(await encryption.crossSigning.isCached()) &&
     341            4 :         !client.isUnknownSession) {
     342            2 :       setState(KeyVerificationState.askSSSS);
     343            2 :       _nextAction = 'request';
     344              :     } else {
     345            3 :       await sendRequest();
     346              :     }
     347              :   }
     348              : 
     349              :   bool _handlePayloadLock = false;
     350              : 
     351            2 :   QRMode getOurQRMode() {
     352              :     QRMode mode = QRMode.verifyOtherUser;
     353            8 :     if (client.userID == userId) {
     354            2 :       if (client.encryption != null &&
     355            3 :           client.encryption!.enabled &&
     356            7 :           (client.userDeviceKeys[client.userID]?.masterKey?.directVerified ??
     357              :               false)) {
     358              :         mode = QRMode.verifySelfTrusted;
     359              :       } else {
     360              :         mode = QRMode.verifySelfUntrusted;
     361              :       }
     362              :     }
     363              :     return mode;
     364              :   }
     365              : 
     366            2 :   Future<void> handlePayload(
     367              :     String type,
     368              :     Map<String, dynamic> payload, [
     369              :     String? eventId,
     370              :   ]) async {
     371            2 :     if (isDone) {
     372              :       return; // no need to do anything with already canceled requests
     373              :     }
     374            2 :     while (_handlePayloadLock) {
     375            0 :       await Future.delayed(Duration(milliseconds: 50));
     376              :     }
     377            2 :     _handlePayloadLock = true;
     378            6 :     Logs().i('[Key Verification] Received type $type: $payload');
     379              :     try {
     380            2 :       var thisLastStep = lastStep;
     381              :       switch (type) {
     382            2 :         case EventTypes.KeyVerificationRequest:
     383            4 :           _deviceId ??= payload['from_device'];
     384            3 :           transactionId ??= eventId ?? payload['transaction_id'];
     385              :           // verify the timestamp
     386            2 :           final now = DateTime.now();
     387              :           final verifyTime =
     388            4 :               DateTime.fromMillisecondsSinceEpoch(payload['timestamp']);
     389            6 :           if (now.subtract(Duration(minutes: 10)).isAfter(verifyTime) ||
     390            6 :               now.add(Duration(minutes: 5)).isBefore(verifyTime)) {
     391              :             // if the request is more than 20min in the past we just silently fail it
     392              :             // to not generate too many cancels
     393            0 :             await cancel(
     394              :               'm.timeout',
     395            0 :               now.subtract(Duration(minutes: 20)).isAfter(verifyTime),
     396              :             );
     397              :             return;
     398              :           }
     399              : 
     400              :           // ensure we have the other sides keys
     401           14 :           if (client.userDeviceKeys[userId]?.deviceKeys[deviceId!] == null) {
     402            0 :             await client.updateUserDeviceKeys(additionalUsers: {userId});
     403            0 :             if (client.userDeviceKeys[userId]?.deviceKeys[deviceId!] == null) {
     404            0 :               await cancel('im.fluffychat.unknown_device');
     405              :               return;
     406              :             }
     407              :           }
     408              : 
     409            6 :           oppositePossibleMethods = List<String>.from(payload['methods']);
     410              :           // verify it has a method we can use
     411            4 :           possibleMethods = _calculatePossibleMethods(
     412            2 :             knownVerificationMethods,
     413            2 :             payload['methods'],
     414              :           );
     415            4 :           if (possibleMethods.isEmpty) {
     416              :             // reject it outright
     417            0 :             await cancel('m.unknown_method');
     418              :             return;
     419              :           }
     420              : 
     421            2 :           setState(KeyVerificationState.askAccept);
     422              :           break;
     423            2 :         case EventTypes.KeyVerificationReady:
     424            4 :           if (deviceId == '*') {
     425            2 :             _deviceId = payload['from_device']; // gotta set the real device id
     426            1 :             transactionId ??= eventId ?? payload['transaction_id'];
     427              :             // and broadcast the cancel to the other devices
     428            1 :             final devices = List<DeviceKeys>.from(
     429            6 :               client.userDeviceKeys[userId]?.deviceKeys.values ??
     430            0 :                   Iterable.empty(),
     431              :             );
     432            1 :             devices.removeWhere(
     433            6 :               (d) => {deviceId, client.deviceID}.contains(d.deviceId),
     434              :             );
     435            1 :             final cancelPayload = <String, dynamic>{
     436              :               'reason': 'Another device accepted the request',
     437              :               'code': 'm.accepted',
     438              :             };
     439            1 :             makePayload(cancelPayload);
     440            2 :             await client.sendToDeviceEncrypted(
     441              :               devices,
     442              :               EventTypes.KeyVerificationCancel,
     443              :               cancelPayload,
     444              :             );
     445              :           }
     446            3 :           _deviceId ??= payload['from_device'];
     447              : 
     448              :           // ensure we have the other sides keys
     449           14 :           if (client.userDeviceKeys[userId]?.deviceKeys[deviceId!] == null) {
     450            0 :             await client.updateUserDeviceKeys(additionalUsers: {userId});
     451            0 :             if (client.userDeviceKeys[userId]?.deviceKeys[deviceId!] == null) {
     452            0 :               await cancel('im.fluffychat.unknown_device');
     453              :               return;
     454              :             }
     455              :           }
     456              : 
     457            6 :           oppositePossibleMethods = List<String>.from(payload['methods']);
     458            4 :           possibleMethods = _calculatePossibleMethods(
     459            2 :             knownVerificationMethods,
     460            2 :             payload['methods'],
     461              :           );
     462            4 :           if (possibleMethods.isEmpty) {
     463              :             // reject it outright
     464            0 :             await cancel('m.unknown_method');
     465              :             return;
     466              :           }
     467              :           // as both parties can send a start, the last step being "ready" is race-condition prone
     468              :           // as such, we better set it *before* we send our start
     469            2 :           lastStep = type;
     470              : 
     471              :           // setup QRData from outgoing request (incoming ready)
     472            4 :           qrCode = await generateQrCode();
     473              : 
     474              :           // play nice with sdks < 0.20.5
     475              :           // https://matrix.to/#/!KBwfdofYJUmnsVoqwn:famedly.de/$wlHXlLQJdfrqKAF5KkuQrXydwOhY_uyqfH4ReasZqnA?via=neko.dev&via=famedly.de&via=lihotzki.de
     476            6 :           if (!isQrSupported(knownVerificationMethods, payload['methods'])) {
     477            4 :             if (knownVerificationMethods.contains(EventTypes.Sas)) {
     478            2 :               final method = _method =
     479            6 :                   _makeVerificationMethod(possibleMethods.first, this);
     480            2 :               await method.sendStart();
     481            2 :               setState(KeyVerificationState.waitingAccept);
     482              :             }
     483              :           } else {
     484              :             // allow user to choose
     485            2 :             setState(KeyVerificationState.askChoice);
     486              :           }
     487              : 
     488              :           break;
     489            2 :         case EventTypes.KeyVerificationStart:
     490            2 :           _deviceId ??= payload['from_device'];
     491            2 :           transactionId ??= eventId ?? payload['transaction_id'];
     492            2 :           if (_method != null) {
     493              :             // the other side sent us a start, even though we already sent one
     494            0 :             if (payload['method'] == _method!.type) {
     495              :               // same method. Determine priority
     496            0 :               final ourEntry = '${client.userID}|${client.deviceID}';
     497            0 :               final entries = [ourEntry, '$userId|$deviceId'];
     498            0 :               entries.sort();
     499            0 :               if (entries.first == ourEntry) {
     500              :                 // our start won, nothing to do
     501              :                 return;
     502              :               } else {
     503              :                 // the other start won, let's hand off
     504            0 :                 startedVerification = false; // it is now as if they started
     505            0 :                 thisLastStep = lastStep =
     506              :                     EventTypes.KeyVerificationRequest; // we fake the last step
     507            0 :                 _method!.dispose(); // in case anything got created already
     508              :               }
     509              :             } else {
     510              :               // methods don't match up, let's cancel this
     511            0 :               await cancel('m.unexpected_message');
     512              :               return;
     513              :             }
     514              :           }
     515            4 :           if (!(await verifyLastStep([
     516              :             EventTypes.KeyVerificationRequest,
     517              :             EventTypes.KeyVerificationReady,
     518              :           ]))) {
     519              :             return; // abort
     520              :           }
     521            6 :           if (!knownVerificationMethods.contains(payload['method'])) {
     522            0 :             await cancel('m.unknown_method');
     523              :             return;
     524              :           }
     525              : 
     526            4 :           if (lastStep == EventTypes.KeyVerificationRequest) {
     527            6 :             if (!possibleMethods.contains(payload['method'])) {
     528            1 :               await cancel('m.unknown_method');
     529              :               return;
     530              :             }
     531              :           }
     532              : 
     533              :           // ensure we have the other sides keys
     534           14 :           if (client.userDeviceKeys[userId]?.deviceKeys[deviceId!] == null) {
     535            0 :             await client.updateUserDeviceKeys(additionalUsers: {userId});
     536            0 :             if (client.userDeviceKeys[userId]?.deviceKeys[deviceId!] == null) {
     537            0 :               await cancel('im.fluffychat.unknown_device');
     538              :               return;
     539              :             }
     540              :           }
     541              : 
     542            6 :           _method = _makeVerificationMethod(payload['method'], this);
     543            2 :           if (lastStep == null) {
     544              :             // validate the start time
     545            0 :             if (room != null) {
     546              :               // we just silently ignore in-room-verification starts
     547            0 :               await cancel('m.unknown_method', true);
     548              :               return;
     549              :             }
     550              :             // validate the specific payload
     551            0 :             if (!_method!.validateStart(payload)) {
     552            0 :               await cancel('m.unknown_method');
     553              :               return;
     554              :             }
     555            0 :             startPayload = payload;
     556            0 :             setState(KeyVerificationState.askAccept);
     557              :           } else {
     558            4 :             Logs().i('handling start in method.....');
     559            4 :             await _method!.handlePayload(type, payload);
     560              :           }
     561              :           break;
     562            2 :         case EventTypes.KeyVerificationDone:
     563            4 :           if (state == KeyVerificationState.showQRSuccess) {
     564            4 :             await send(EventTypes.KeyVerificationDone, {});
     565            2 :             setState(KeyVerificationState.done);
     566              :           }
     567              :           break;
     568            1 :         case EventTypes.KeyVerificationCancel:
     569            1 :           canceled = true;
     570            2 :           canceledCode = payload['code'];
     571            2 :           canceledReason = payload['reason'];
     572            1 :           setState(KeyVerificationState.error);
     573              :           break;
     574              :         default:
     575            1 :           final method = _method;
     576              :           if (method != null) {
     577            1 :             await method.handlePayload(type, payload);
     578              :           } else {
     579            0 :             await cancel('m.invalid_message');
     580              :           }
     581              :           break;
     582              :       }
     583            4 :       if (lastStep == thisLastStep) {
     584            2 :         lastStep = type;
     585              :       }
     586              :     } catch (err, stacktrace) {
     587            0 :       Logs().e('[Key Verification] An error occured', err, stacktrace);
     588            0 :       await cancel('m.invalid_message');
     589              :     } finally {
     590            2 :       _handlePayloadLock = false;
     591              :     }
     592              :   }
     593              : 
     594            1 :   void otherDeviceAccepted() {
     595            1 :     canceled = true;
     596            1 :     canceledCode = 'm.accepted';
     597            1 :     canceledReason = 'm.accepted';
     598            1 :     setState(KeyVerificationState.error);
     599              :   }
     600              : 
     601            2 :   Future<void> openSSSS({
     602              :     String? passphrase,
     603              :     String? recoveryKey,
     604              :     String? keyOrPassphrase,
     605              :     bool skip = false,
     606              :   }) async {
     607            2 :     Future<void> next() async {
     608            4 :       if (_nextAction == 'request') {
     609            2 :         await sendRequest();
     610            4 :       } else if (_nextAction == 'done') {
     611              :         // and now let's sign them all in the background
     612           10 :         unawaited(encryption.crossSigning.sign(_verifiedDevices));
     613            2 :         setState(KeyVerificationState.done);
     614            0 :       } else if (_nextAction == 'showQRSuccess') {
     615            0 :         setState(KeyVerificationState.showQRSuccess);
     616              :       }
     617              :     }
     618              : 
     619              :     if (skip) {
     620            0 :       await next();
     621              :       return;
     622              :     }
     623            6 :     final handle = encryption.ssss.open(EventTypes.CrossSigningUserSigning);
     624            2 :     await handle.unlock(
     625              :       passphrase: passphrase,
     626              :       recoveryKey: recoveryKey,
     627              :       keyOrPassphrase: keyOrPassphrase,
     628              :     );
     629            2 :     await handle.maybeCacheAll();
     630            2 :     await next();
     631              :   }
     632              : 
     633              :   /// called when the user accepts an incoming verification
     634            2 :   Future<void> acceptVerification() async {
     635            4 :     if (!(await verifyLastStep([
     636              :       EventTypes.KeyVerificationRequest,
     637              :       EventTypes.KeyVerificationStart,
     638              :     ]))) {
     639              :       return;
     640              :     }
     641            2 :     setState(KeyVerificationState.waitingAccept);
     642            4 :     if (lastStep == EventTypes.KeyVerificationRequest) {
     643              :       final copyKnownVerificationMethods =
     644            4 :           List<String>.from(knownVerificationMethods);
     645              :       // qr code only works when atleast one side has verified master key
     646            8 :       if (userId == client.userID) {
     647            8 :         if (!(client.userDeviceKeys[client.userID]?.deviceKeys[deviceId]
     648            1 :                     ?.hasValidSignatureChain(verifiedByTheirMasterKey: true) ??
     649              :                 false) &&
     650            7 :             !(client.userDeviceKeys[client.userID]?.masterKey?.verified ??
     651              :                 false)) {
     652              :           copyKnownVerificationMethods
     653            3 :               .removeWhere((element) => element.startsWith('m.qr_code'));
     654            1 :           copyKnownVerificationMethods.remove(EventTypes.Reciprocate);
     655              : 
     656              :           // we are removing stuff only using the old possibleMethods should be ok here.
     657            2 :           final copyPossibleMethods = List<String>.from(possibleMethods);
     658            2 :           possibleMethods = _calculatePossibleMethods(
     659              :             copyKnownVerificationMethods,
     660              :             copyPossibleMethods,
     661              :           );
     662              :         }
     663              :       }
     664              :       // we need to send a ready event
     665            4 :       await send(EventTypes.KeyVerificationReady, {
     666              :         'methods': copyKnownVerificationMethods,
     667              :       });
     668              :       // setup QRData from incoming request (outgoing ready)
     669            4 :       qrCode = await generateQrCode();
     670            2 :       setState(KeyVerificationState.askChoice);
     671              :     } else {
     672              :       // we need to send an accept event
     673            0 :       await _method!
     674            0 :           .handlePayload(EventTypes.KeyVerificationStart, startPayload!);
     675              :     }
     676              :   }
     677              : 
     678              :   /// called when the user rejects an incoming verification
     679            1 :   Future<void> rejectVerification() async {
     680            1 :     if (isDone) {
     681              :       return;
     682              :     }
     683            2 :     if (!(await verifyLastStep([
     684              :       EventTypes.KeyVerificationRequest,
     685              :       EventTypes.KeyVerificationStart,
     686              :     ]))) {
     687              :       return;
     688              :     }
     689            1 :     await cancel('m.user');
     690              :   }
     691              : 
     692              :   /// call this to confirm that your other device has shown a shield and is in
     693              :   /// `done` state.
     694            2 :   Future<void> acceptQRScanConfirmation() async {
     695            4 :     if (_method is _KeyVerificationMethodQRReciprocate &&
     696            4 :         state == KeyVerificationState.confirmQRScan) {
     697            2 :       await (_method as _KeyVerificationMethodQRReciprocate)
     698            2 :           .acceptQRScanConfirmation();
     699              :     }
     700              :   }
     701              : 
     702            1 :   Future<void> acceptSas() async {
     703            2 :     if (_method is _KeyVerificationMethodSas) {
     704            2 :       await (_method as _KeyVerificationMethodSas).acceptSas();
     705              :     }
     706              :   }
     707              : 
     708            1 :   Future<void> rejectSas() async {
     709            2 :     if (_method is _KeyVerificationMethodSas) {
     710            2 :       await (_method as _KeyVerificationMethodSas).rejectSas();
     711              :     }
     712              :   }
     713              : 
     714            1 :   List<int> get sasNumbers {
     715            2 :     if (_method is _KeyVerificationMethodSas) {
     716            3 :       return _bytesToInt((_method as _KeyVerificationMethodSas).makeSas(5), 13)
     717            3 :           .map((n) => n + 1000)
     718            1 :           .toList();
     719              :     }
     720            0 :     return [];
     721              :   }
     722              : 
     723            1 :   List<String> get sasTypes {
     724            2 :     if (_method is _KeyVerificationMethodSas) {
     725            2 :       return (_method as _KeyVerificationMethodSas).authenticationTypes ?? [];
     726              :     }
     727            0 :     return [];
     728              :   }
     729              : 
     730            1 :   List<KeyVerificationEmoji> get sasEmojis {
     731            2 :     if (_method is _KeyVerificationMethodSas) {
     732              :       final numbers =
     733            3 :           _bytesToInt((_method as _KeyVerificationMethodSas).makeSas(6), 6);
     734            5 :       return numbers.map((n) => KeyVerificationEmoji(n)).toList().sublist(0, 7);
     735              :     }
     736            0 :     return [];
     737              :   }
     738              : 
     739            1 :   Future<void> maybeRequestSSSSSecrets([int i = 0]) async {
     740            1 :     final requestInterval = <int>[10, 60];
     741            3 :     if ((!encryption.crossSigning.enabled ||
     742            3 :             (encryption.crossSigning.enabled &&
     743            3 :                 (await encryption.crossSigning.isCached()))) &&
     744            0 :         (!encryption.keyManager.enabled ||
     745            0 :             (encryption.keyManager.enabled &&
     746            0 :                 (await encryption.keyManager.isCached())))) {
     747              :       // no need to request cache, we already have it
     748              :       return;
     749              :     }
     750              :     // ignore: unawaited_futures
     751            2 :     encryption.ssss
     752            4 :         .maybeRequestAll(_verifiedDevices.whereType<DeviceKeys>().toList());
     753            2 :     if (requestInterval.length <= i) {
     754              :       return;
     755              :     }
     756            1 :     Timer(
     757            2 :       Duration(seconds: requestInterval[i]),
     758            0 :       () => maybeRequestSSSSSecrets(i + 1),
     759              :     );
     760              :   }
     761              : 
     762            1 :   Future<void> verifyKeysSAS(
     763              :     Map<String, String> keys,
     764              :     Future<bool> Function(String, SignableKey) verifier,
     765              :   ) async {
     766            2 :     _verifiedDevices = <SignableKey>[];
     767              : 
     768            4 :     final userDeviceKey = client.userDeviceKeys[userId];
     769              :     if (userDeviceKey == null) {
     770            0 :       await cancel('m.key_mismatch');
     771              :       return;
     772              :     }
     773            2 :     for (final entry in keys.entries) {
     774            1 :       final keyId = entry.key;
     775            2 :       final verifyDeviceId = keyId.substring('ed25519:'.length);
     776            1 :       final keyInfo = entry.value;
     777            1 :       final key = userDeviceKey.getKey(verifyDeviceId);
     778              :       if (key != null) {
     779            1 :         if (!(await verifier(keyInfo, key))) {
     780            0 :           await cancel('m.key_mismatch');
     781              :           return;
     782              :         }
     783            2 :         _verifiedDevices.add(key);
     784              :       }
     785              :     }
     786              :     // okay, we reached this far, so all the devices are verified!
     787              :     var verifiedMasterKey = false;
     788            2 :     final wasUnknownSession = client.isUnknownSession;
     789            2 :     for (final key in _verifiedDevices) {
     790            1 :       await key.setVerified(
     791              :         true,
     792              :         false,
     793              :       ); // we don't want to sign the keys juuuust yet
     794            3 :       if (key is CrossSigningKey && key.usage.contains('master')) {
     795              :         verifiedMasterKey = true;
     796              :       }
     797              :     }
     798            4 :     if (verifiedMasterKey && userId == client.userID) {
     799              :       // it was our own master key, let's request the cross signing keys
     800              :       // we do it in the background, thus no await needed here
     801              :       // ignore: unawaited_futures
     802            0 :       maybeRequestSSSSSecrets();
     803              :     }
     804            2 :     await send(EventTypes.KeyVerificationDone, {});
     805              : 
     806              :     var askingSSSS = false;
     807            3 :     if (encryption.crossSigning.enabled &&
     808            4 :         encryption.crossSigning.signable(_verifiedDevices)) {
     809              :       // these keys can be signed! Let's do so
     810            3 :       if (await encryption.crossSigning.isCached()) {
     811              :         // we want to make sure the verification state is correct for the other party after this event is handled.
     812              :         // Otherwise the verification dialog might be stuck in an unverified but done state for a bit.
     813            0 :         await encryption.crossSigning.sign(_verifiedDevices);
     814              :       } else if (!wasUnknownSession) {
     815              :         askingSSSS = true;
     816              :       }
     817              :     }
     818              :     if (askingSSSS) {
     819            1 :       setState(KeyVerificationState.askSSSS);
     820            1 :       _nextAction = 'done';
     821              :     } else {
     822            1 :       setState(KeyVerificationState.done);
     823              :     }
     824              :   }
     825              : 
     826              :   /// shower is true only for reciprocated verifications (shower side)
     827            2 :   Future<void> verifyKeysQR(SignableKey key, {bool shower = true}) async {
     828              :     var verifiedMasterKey = false;
     829            4 :     final wasUnknownSession = client.isUnknownSession;
     830              : 
     831            2 :     key.setDirectVerified(true);
     832            6 :     if (key is CrossSigningKey && key.usage.contains('master')) {
     833              :       verifiedMasterKey = true;
     834              :     }
     835              : 
     836            8 :     if (verifiedMasterKey && userId == client.userID) {
     837              :       // it was our own master key, let's request the cross signing keys
     838              :       // we do it in the background, thus no await needed here
     839              :       // ignore: unawaited_futures
     840            1 :       maybeRequestSSSSSecrets();
     841              :     }
     842              :     if (shower) {
     843            4 :       await send(EventTypes.KeyVerificationDone, {});
     844              :     }
     845            4 :     final keyList = List<SignableKey>.from([key]);
     846              :     var askingSSSS = false;
     847            6 :     if (encryption.crossSigning.enabled &&
     848            6 :         encryption.crossSigning.signable(keyList)) {
     849              :       // these keys can be signed! Let's do so
     850            6 :       if (await encryption.crossSigning.isCached()) {
     851              :         // we want to make sure the verification state is correct for the other party after this event is handled.
     852              :         // Otherwise the verification dialog might be stuck in an unverified but done state for a bit.
     853            6 :         await encryption.crossSigning.sign(keyList);
     854              :       } else if (!wasUnknownSession) {
     855              :         askingSSSS = true;
     856              :       }
     857              :     }
     858              :     if (askingSSSS) {
     859              :       // no need to worry about shower/scanner here because if scanner was
     860              :       // verified, ssss is already
     861            1 :       setState(KeyVerificationState.askSSSS);
     862              :       if (shower) {
     863            1 :         _nextAction = 'done';
     864              :       } else {
     865            0 :         _nextAction = 'showQRSuccess';
     866              :       }
     867              :     } else {
     868              :       if (shower) {
     869            2 :         setState(KeyVerificationState.done);
     870              :       } else {
     871            2 :         setState(KeyVerificationState.showQRSuccess);
     872              :       }
     873              :     }
     874              :   }
     875              : 
     876            2 :   Future<bool> verifyActivity() async {
     877           10 :     if (lastActivity.add(Duration(minutes: 10)).isAfter(DateTime.now())) {
     878            4 :       lastActivity = DateTime.now();
     879              :       return true;
     880              :     }
     881            0 :     await cancel('m.timeout');
     882              :     return false;
     883              :   }
     884              : 
     885            2 :   Future<bool> verifyLastStep(List<String?> checkLastStep) async {
     886            2 :     if (!(await verifyActivity())) {
     887              :       return false;
     888              :     }
     889            4 :     if (checkLastStep.contains(lastStep)) {
     890              :       return true;
     891              :     }
     892            0 :     Logs().e(
     893            0 :       '[KeyVerificaton] lastStep mismatch cancelling, expected from ${checkLastStep.toString()} was ${lastStep.toString()}',
     894              :     );
     895            0 :     await cancel('m.unexpected_message');
     896              :     return false;
     897              :   }
     898              : 
     899            2 :   Future<void> cancel([String code = 'm.unknown', bool quiet = false]) async {
     900            3 :     if (!quiet && (deviceId != null || room != null)) {
     901            4 :       await send(EventTypes.KeyVerificationCancel, {
     902              :         'reason': code,
     903              :         'code': code,
     904              :       });
     905              :     }
     906            2 :     canceled = true;
     907            2 :     canceledCode = code;
     908            2 :     setState(KeyVerificationState.error);
     909              :   }
     910              : 
     911            3 :   void makePayload(Map<String, dynamic> payload) {
     912            9 :     payload['from_device'] = client.deviceID;
     913            3 :     if (transactionId != null) {
     914            3 :       if (room != null) {
     915            2 :         payload['m.relates_to'] = {
     916              :           'rel_type': 'm.reference',
     917            1 :           'event_id': transactionId,
     918              :         };
     919              :       } else {
     920            4 :         payload['transaction_id'] = transactionId;
     921              :       }
     922              :     }
     923              :   }
     924              : 
     925            3 :   Future<void> send(
     926              :     String type,
     927              :     Map<String, dynamic> payload,
     928              :   ) async {
     929            3 :     makePayload(payload);
     930            9 :     Logs().i('[Key Verification] Sending type $type: $payload');
     931            3 :     if (room != null) {
     932           12 :       Logs().i('[Key Verification] Sending to $userId in room ${room!.id}...');
     933            4 :       if ({EventTypes.KeyVerificationRequest}.contains(type)) {
     934            2 :         payload['msgtype'] = type;
     935            4 :         payload['to'] = userId;
     936            2 :         payload['body'] =
     937            2 :             'Attempting verification request. ($type) Apparently your client doesn\'t support this';
     938              :         type = EventTypes.Message;
     939              :       }
     940            4 :       final newTransactionId = await room!.sendEvent(payload, type: type);
     941            2 :       if (transactionId == null) {
     942            2 :         transactionId = newTransactionId;
     943            6 :         encryption.keyVerificationManager.addRequest(this);
     944              :       }
     945              :     } else {
     946           10 :       Logs().i('[Key Verification] Sending to $userId device $deviceId...');
     947            4 :       if (deviceId == '*') {
     948              :         if ({
     949            1 :           EventTypes.KeyVerificationRequest,
     950            1 :           EventTypes.KeyVerificationCancel,
     951            1 :         }.contains(type)) {
     952              :           final deviceKeys =
     953            7 :               client.userDeviceKeys[userId]?.deviceKeys.values.where(
     954            2 :             (deviceKey) => deviceKey.hasValidSignatureChain(
     955              :               verifiedByTheirMasterKey: true,
     956              :             ),
     957              :           );
     958              : 
     959              :           if (deviceKeys != null) {
     960            2 :             await client.sendToDeviceEncrypted(
     961            1 :               deviceKeys.toList(),
     962              :               type,
     963              :               payload,
     964              :             );
     965              :           }
     966              :         } else {
     967            0 :           Logs().e(
     968            0 :             '[Key Verification] Tried to broadcast and un-broadcastable type: $type',
     969              :           );
     970              :         }
     971              :       } else {
     972           14 :         if (client.userDeviceKeys[userId]?.deviceKeys[deviceId] != null) {
     973            4 :           await client.sendToDeviceEncrypted(
     974           16 :             [client.userDeviceKeys[userId]!.deviceKeys[deviceId]!],
     975              :             type,
     976              :             payload,
     977              :           );
     978              :         } else {
     979            0 :           Logs().e('[Key Verification] Unknown device');
     980              :         }
     981              :       }
     982              :     }
     983              :   }
     984              : 
     985            3 :   void setState(KeyVerificationState newState) {
     986            6 :     if (state != KeyVerificationState.error) {
     987            3 :       state = newState;
     988              :     }
     989              : 
     990            3 :     onUpdate?.call();
     991              :   }
     992              : 
     993              :   static const String prefix = 'MATRIX';
     994              :   static const int version = 0x02;
     995              : 
     996            2 :   Future<bool> verifyQrData(Uint8List qrDataRawBytes) async {
     997              :     final data = qrDataRawBytes;
     998              :     // hardcoded stuff + 2 keys + secret
     999           18 :     if (data.length < 10 + 32 + 32 + 8 + utf8.encode(transactionId!).length) {
    1000              :       return false;
    1001              :     }
    1002            4 :     if (data[6] != version) return false;
    1003              :     final remoteQrMode =
    1004           10 :         QRMode.values.singleWhere((mode) => mode.code == data[7]);
    1005            6 :     if (ascii.decode(data.sublist(0, 6)) != prefix) return false;
    1006            4 :     if (data[6] != version) return false;
    1007            8 :     final tmpBuf = Uint8List.fromList([data[8], data[9]]);
    1008            6 :     final encodedTxnLen = ByteData.view(tmpBuf.buffer).getUint16(0);
    1009           10 :     if (utf8.decode(data.sublist(10, 10 + encodedTxnLen)) != transactionId) {
    1010              :       return false;
    1011              :     }
    1012            4 :     final keys = client.userDeviceKeys;
    1013              : 
    1014            6 :     final ownKeys = keys[client.userID];
    1015            4 :     final otherUserKeys = keys[userId];
    1016            2 :     final ownMasterKey = ownKeys?.getCrossSigningKey('master');
    1017            6 :     final ownDeviceKey = ownKeys?.getKey(client.deviceID!);
    1018            4 :     final ownOtherDeviceKey = ownKeys?.getKey(deviceId!);
    1019            2 :     final otherUserMasterKey = otherUserKeys?.masterKey;
    1020              : 
    1021            2 :     final secondKey = encodeBase64Unpadded(
    1022           12 :       data.sublist(10 + encodedTxnLen + 32, 10 + encodedTxnLen + 32 + 32),
    1023              :     );
    1024              :     final randomSharedSecret =
    1025           10 :         encodeBase64Unpadded(data.sublist(10 + encodedTxnLen + 32 + 32));
    1026              : 
    1027              :     /// `request.randomSharedSecretForQRCode` is overwritten below to send with `sendStart`
    1028            4 :     if ({QRMode.verifyOtherUser, QRMode.verifySelfUntrusted}
    1029            2 :         .contains(remoteQrMode)) {
    1030            2 :       if (!(ownMasterKey?.verified ?? false)) {
    1031            0 :         Logs().e(
    1032              :           '[KeyVerification] verifyQrData because you were in mode 0/2 and had untrusted msk',
    1033              :         );
    1034              :         return false;
    1035              :       }
    1036              :     }
    1037              : 
    1038            2 :     if (remoteQrMode == QRMode.verifyOtherUser &&
    1039              :         otherUserMasterKey != null &&
    1040              :         ownMasterKey != null) {
    1041            2 :       if (secondKey == ownMasterKey.ed25519Key) {
    1042            1 :         randomSharedSecretForQRCode = randomSharedSecret;
    1043            1 :         await verifyKeysQR(otherUserMasterKey, shower: false);
    1044              :         return true;
    1045              :       }
    1046            1 :     } else if (remoteQrMode == QRMode.verifySelfTrusted &&
    1047              :         ownMasterKey != null &&
    1048              :         ownDeviceKey != null) {
    1049            2 :       if (secondKey == ownDeviceKey.ed25519Key) {
    1050            1 :         randomSharedSecretForQRCode = randomSharedSecret;
    1051            1 :         await verifyKeysQR(ownMasterKey, shower: false);
    1052              :         return true;
    1053              :       }
    1054            1 :     } else if (remoteQrMode == QRMode.verifySelfUntrusted &&
    1055              :         ownOtherDeviceKey != null &&
    1056              :         ownMasterKey != null) {
    1057            2 :       if (secondKey == ownMasterKey.ed25519Key) {
    1058            1 :         randomSharedSecretForQRCode = randomSharedSecret;
    1059            1 :         await verifyKeysQR(ownOtherDeviceKey, shower: false);
    1060              :         return true;
    1061              :       }
    1062              :     }
    1063              : 
    1064              :     return false;
    1065              :   }
    1066              : 
    1067            2 :   Future<(String, String)?> getKeys(QRMode mode) async {
    1068            4 :     final keys = client.userDeviceKeys;
    1069              : 
    1070            6 :     final ownKeys = keys[client.userID];
    1071            4 :     final otherUserKeys = keys[userId];
    1072            6 :     final ownDeviceKey = ownKeys?.getKey(client.deviceID!);
    1073            2 :     final ownMasterKey = ownKeys?.getCrossSigningKey('master');
    1074            4 :     final otherDeviceKey = otherUserKeys?.getKey(deviceId!);
    1075            2 :     final otherMasterKey = otherUserKeys?.getCrossSigningKey('master');
    1076              : 
    1077            2 :     if (mode == QRMode.verifyOtherUser &&
    1078              :         ownMasterKey != null &&
    1079              :         otherMasterKey != null) {
    1080              :       // we already have this check when sending `knownVerificationMethods`, but
    1081              :       // just to be safe anyway
    1082            1 :       if (ownMasterKey.verified) {
    1083            2 :         return (ownMasterKey.ed25519Key!, otherMasterKey.ed25519Key!);
    1084              :       }
    1085            1 :     } else if (mode == QRMode.verifySelfTrusted &&
    1086              :         ownMasterKey != null &&
    1087              :         otherDeviceKey != null) {
    1088            1 :       if (ownMasterKey.verified) {
    1089            2 :         return (ownMasterKey.ed25519Key!, otherDeviceKey.ed25519Key!);
    1090              :       }
    1091            1 :     } else if (mode == QRMode.verifySelfUntrusted &&
    1092              :         ownMasterKey != null &&
    1093              :         ownDeviceKey != null) {
    1094            2 :       return (ownDeviceKey.ed25519Key!, ownMasterKey.ed25519Key!);
    1095              :     }
    1096              :     return null;
    1097              :   }
    1098              : 
    1099            2 :   Future<QRCode?> generateQrCode() async {
    1100            2 :     final data = Uint8Buffer();
    1101              :     // why 11? https://github.com/matrix-org/matrix-js-sdk/commit/275ea6aacbfc6623e7559a7649ca5cab207903d9
    1102            2 :     randomSharedSecretForQRCode =
    1103            4 :         encodeBase64Unpadded(uc.secureRandomBytes(11));
    1104              : 
    1105            2 :     final mode = getOurQRMode();
    1106            4 :     data.addAll(ascii.encode(prefix));
    1107            2 :     data.add(version);
    1108            4 :     data.add(mode.code);
    1109            4 :     final encodedTxnId = utf8.encode(transactionId!);
    1110            2 :     final txnIdLen = encodedTxnId.length;
    1111            2 :     final tmpBuf = Uint8List(2);
    1112            6 :     ByteData.view(tmpBuf.buffer).setUint16(0, txnIdLen);
    1113            2 :     data.addAll(tmpBuf);
    1114            2 :     data.addAll(encodedTxnId);
    1115            2 :     final keys = await getKeys(mode);
    1116              :     if (keys != null) {
    1117            4 :       data.addAll(base64decodeUnpadded(keys.$1));
    1118            4 :       data.addAll(base64decodeUnpadded(keys.$2));
    1119              :     } else {
    1120              :       return null;
    1121              :     }
    1122              : 
    1123            6 :     data.addAll(base64decodeUnpadded(randomSharedSecretForQRCode!));
    1124            4 :     return QRCode(randomSharedSecretForQRCode!, data);
    1125              :   }
    1126              : }
    1127              : 
    1128              : abstract class _KeyVerificationMethod {
    1129              :   KeyVerification request;
    1130            3 :   Encryption get encryption => request.encryption;
    1131            6 :   Client get client => request.client;
    1132            2 :   _KeyVerificationMethod({required this.request});
    1133              : 
    1134              :   Future<void> handlePayload(String type, Map<String, dynamic> payload);
    1135            0 :   bool validateStart(Map<String, dynamic> payload) {
    1136              :     return false;
    1137              :   }
    1138              : 
    1139              :   late String _type;
    1140            4 :   String get type => _type;
    1141              : 
    1142              :   Future<void> sendStart();
    1143            0 :   void dispose() {}
    1144              : }
    1145              : 
    1146              : class _KeyVerificationMethodQRReciprocate extends _KeyVerificationMethod {
    1147            2 :   _KeyVerificationMethodQRReciprocate({required super.request});
    1148              : 
    1149              :   @override
    1150              :   // ignore: overridden_fields
    1151              :   final _type = EventTypes.Reciprocate;
    1152              : 
    1153            2 :   @override
    1154              :   bool validateStart(Map<String, dynamic> payload) {
    1155            6 :     if (payload['method'] != type) return false;
    1156            8 :     if (payload['secret'] != request.randomSharedSecretForQRCode) return false;
    1157              :     return true;
    1158              :   }
    1159              : 
    1160            2 :   @override
    1161              :   Future<void> handlePayload(String type, Map<String, dynamic> payload) async {
    1162              :     try {
    1163              :       switch (type) {
    1164            2 :         case EventTypes.KeyVerificationStart:
    1165            6 :           if (!(await request.verifyLastStep([
    1166              :             EventTypes.KeyVerificationReady,
    1167              :             EventTypes.KeyVerificationRequest,
    1168              :           ]))) {
    1169              :             return; // abort
    1170              :           }
    1171            2 :           if (!validateStart(payload)) {
    1172            2 :             await request.cancel('m.invalid_message');
    1173              :             return;
    1174              :           }
    1175            4 :           request.setState(KeyVerificationState.confirmQRScan);
    1176              :           break;
    1177              :       }
    1178              :     } catch (e, s) {
    1179            0 :       Logs().e('[Key Verification Reciprocate] An error occured', e, s);
    1180            0 :       if (request.deviceId != null) {
    1181            0 :         await request.cancel('m.invalid_message');
    1182              :       }
    1183              :     }
    1184              :   }
    1185              : 
    1186            2 :   Future<void> acceptQRScanConfirmation() async {
    1187              :     // secret validation already done in validateStart
    1188              : 
    1189            4 :     final ourQRMode = request.getOurQRMode();
    1190              :     SignableKey? keyToVerify;
    1191              : 
    1192            2 :     if (ourQRMode == QRMode.verifyOtherUser) {
    1193            6 :       keyToVerify = client.userDeviceKeys[request.userId]?.masterKey;
    1194            1 :     } else if (ourQRMode == QRMode.verifySelfTrusted) {
    1195              :       keyToVerify =
    1196            9 :           client.userDeviceKeys[client.userID]?.deviceKeys[request.deviceId];
    1197            1 :     } else if (ourQRMode == QRMode.verifySelfUntrusted) {
    1198            6 :       keyToVerify = client.userDeviceKeys[client.userID]?.masterKey;
    1199              :     }
    1200              :     if (keyToVerify != null) {
    1201            4 :       await request.verifyKeysQR(keyToVerify, shower: true);
    1202              :     } else {
    1203            0 :       Logs().e('[KeyVerification], verifying keys failed');
    1204            0 :       await request.cancel('m.invalid_key');
    1205              :     }
    1206              :   }
    1207              : 
    1208            2 :   @override
    1209              :   Future<void> sendStart() async {
    1210            2 :     final payload = <String, dynamic>{
    1211            2 :       'method': type,
    1212            4 :       'secret': request.randomSharedSecretForQRCode,
    1213              :     };
    1214            4 :     request.makePayload(payload);
    1215            4 :     await request.send(EventTypes.KeyVerificationStart, payload);
    1216              :   }
    1217              : 
    1218            2 :   @override
    1219              :   void dispose() {}
    1220              : }
    1221              : 
    1222              : enum QRMode {
    1223              :   verifyOtherUser(0x00),
    1224              :   verifySelfTrusted(0x01),
    1225              :   verifySelfUntrusted(0x02);
    1226              : 
    1227              :   const QRMode(this.code);
    1228              :   final int code;
    1229              : }
    1230              : 
    1231              : class QRCode {
    1232              :   /// You actually never need this when implementing in a client, its just to
    1233              :   /// make tests easier. Just pass `qrDataRawBytes` in `continueVerifcation()`
    1234              :   final String randomSharedSecret;
    1235              :   final Uint8Buffer qrDataRawBytes;
    1236            2 :   QRCode(this.randomSharedSecret, this.qrDataRawBytes);
    1237              : }
    1238              : 
    1239              : const knownKeyAgreementProtocols = ['curve25519-hkdf-sha256', 'curve25519'];
    1240              : const knownHashes = ['sha256'];
    1241              : const knownHashesAuthentificationCodes = ['hkdf-hmac-sha256'];
    1242              : 
    1243              : class _KeyVerificationMethodSas extends _KeyVerificationMethod {
    1244            2 :   _KeyVerificationMethodSas({required super.request});
    1245              : 
    1246              :   @override
    1247              :   // ignore: overridden_fields
    1248              :   final _type = EventTypes.Sas;
    1249              : 
    1250              :   String? keyAgreementProtocol;
    1251              :   String? hash;
    1252              :   String? messageAuthenticationCode;
    1253              :   List<String>? authenticationTypes;
    1254              :   late String startCanonicalJson;
    1255              :   String? commitment;
    1256              :   late String theirPublicKey;
    1257              :   Map<String, dynamic>? macPayload;
    1258              :   olm.SAS? sas;
    1259              : 
    1260            2 :   @override
    1261              :   void dispose() {
    1262            3 :     sas?.free();
    1263              :   }
    1264              : 
    1265            2 :   List<String> get knownAuthentificationTypes {
    1266            2 :     final types = <String>[];
    1267            6 :     if (request.client.verificationMethods
    1268            2 :         .contains(KeyVerificationMethod.emoji)) {
    1269            2 :       types.add('emoji');
    1270              :     }
    1271            6 :     if (request.client.verificationMethods
    1272            2 :         .contains(KeyVerificationMethod.numbers)) {
    1273            2 :       types.add('decimal');
    1274              :     }
    1275              :     return types;
    1276              :   }
    1277              : 
    1278            1 :   @override
    1279              :   Future<void> handlePayload(String type, Map<String, dynamic> payload) async {
    1280              :     try {
    1281              :       switch (type) {
    1282            1 :         case EventTypes.KeyVerificationStart:
    1283            3 :           if (!(await request.verifyLastStep([
    1284              :             EventTypes.KeyVerificationReady,
    1285              :             EventTypes.KeyVerificationRequest,
    1286              :             EventTypes.KeyVerificationStart,
    1287              :           ]))) {
    1288              :             return; // abort
    1289              :           }
    1290            1 :           if (!validateStart(payload)) {
    1291            0 :             await request.cancel('m.unknown_method');
    1292              :             return;
    1293              :           }
    1294            1 :           await _sendAccept();
    1295              :           break;
    1296            1 :         case EventTypes.KeyVerificationAccept:
    1297            3 :           if (!(await request.verifyLastStep([
    1298              :             EventTypes.KeyVerificationReady,
    1299              :             EventTypes.KeyVerificationRequest,
    1300              :           ]))) {
    1301              :             return;
    1302              :           }
    1303            1 :           if (!_handleAccept(payload)) {
    1304            0 :             await request.cancel('m.unknown_method');
    1305              :             return;
    1306              :           }
    1307            1 :           await _sendKey();
    1308              :           break;
    1309            1 :         case 'm.key.verification.key':
    1310            3 :           if (!(await request.verifyLastStep([
    1311              :             EventTypes.KeyVerificationAccept,
    1312              :             EventTypes.KeyVerificationStart,
    1313              :           ]))) {
    1314              :             return;
    1315              :           }
    1316            1 :           _handleKey(payload);
    1317            3 :           if (request.lastStep == EventTypes.KeyVerificationStart) {
    1318              :             // we need to send our key
    1319            1 :             await _sendKey();
    1320              :           } else {
    1321              :             // we already sent our key, time to verify the commitment being valid
    1322            1 :             if (!_validateCommitment()) {
    1323            0 :               await request.cancel('m.mismatched_commitment');
    1324              :               return;
    1325              :             }
    1326              :           }
    1327            2 :           request.setState(KeyVerificationState.askSas);
    1328              :           break;
    1329            1 :         case 'm.key.verification.mac':
    1330            3 :           if (!(await request.verifyLastStep(['m.key.verification.key']))) {
    1331              :             return;
    1332              :           }
    1333            1 :           macPayload = payload;
    1334            3 :           if (request.state == KeyVerificationState.waitingSas) {
    1335            1 :             await _processMac();
    1336              :           }
    1337              :           break;
    1338              :       }
    1339              :     } catch (err, stacktrace) {
    1340            0 :       Logs().e('[Key Verification SAS] An error occured', err, stacktrace);
    1341            0 :       if (request.deviceId != null) {
    1342            0 :         await request.cancel('m.invalid_message');
    1343              :       }
    1344              :     }
    1345              :   }
    1346              : 
    1347            1 :   Future<void> acceptSas() async {
    1348            1 :     await _sendMac();
    1349            2 :     request.setState(KeyVerificationState.waitingSas);
    1350            1 :     if (macPayload != null) {
    1351            1 :       await _processMac();
    1352              :     }
    1353              :   }
    1354              : 
    1355            1 :   Future<void> rejectSas() async {
    1356            2 :     await request.cancel('m.mismatched_sas');
    1357              :   }
    1358              : 
    1359            2 :   @override
    1360              :   Future<void> sendStart() async {
    1361            2 :     final payload = <String, dynamic>{
    1362            2 :       'method': type,
    1363              :       'key_agreement_protocols': knownKeyAgreementProtocols,
    1364              :       'hashes': knownHashes,
    1365              :       'message_authentication_codes': knownHashesAuthentificationCodes,
    1366            2 :       'short_authentication_string': knownAuthentificationTypes,
    1367              :     };
    1368            4 :     request.makePayload(payload);
    1369              :     // We just store the canonical json in here for later verification
    1370            6 :     startCanonicalJson = String.fromCharCodes(canonicalJson.encode(payload));
    1371            4 :     await request.send(EventTypes.KeyVerificationStart, payload);
    1372              :   }
    1373              : 
    1374            1 :   @override
    1375              :   bool validateStart(Map<String, dynamic> payload) {
    1376            3 :     if (payload['method'] != type) {
    1377              :       return false;
    1378              :     }
    1379            1 :     final possibleKeyAgreementProtocols = _intersect(
    1380              :       knownKeyAgreementProtocols,
    1381            1 :       payload['key_agreement_protocols'],
    1382              :     );
    1383            1 :     if (possibleKeyAgreementProtocols.isEmpty) {
    1384              :       return false;
    1385              :     }
    1386            2 :     keyAgreementProtocol = possibleKeyAgreementProtocols.first;
    1387            2 :     final possibleHashes = _intersect(knownHashes, payload['hashes']);
    1388            1 :     if (possibleHashes.isEmpty) {
    1389              :       return false;
    1390              :     }
    1391            2 :     hash = possibleHashes.first;
    1392            1 :     final possibleMessageAuthenticationCodes = _intersect(
    1393              :       knownHashesAuthentificationCodes,
    1394            1 :       payload['message_authentication_codes'],
    1395              :     );
    1396            1 :     if (possibleMessageAuthenticationCodes.isEmpty) {
    1397              :       return false;
    1398              :     }
    1399            2 :     messageAuthenticationCode = possibleMessageAuthenticationCodes.first;
    1400            1 :     final possibleAuthenticationTypes = _intersect(
    1401            1 :       knownAuthentificationTypes,
    1402            1 :       payload['short_authentication_string'],
    1403              :     );
    1404            1 :     if (possibleAuthenticationTypes.isEmpty) {
    1405              :       return false;
    1406              :     }
    1407            1 :     authenticationTypes = possibleAuthenticationTypes;
    1408            3 :     startCanonicalJson = String.fromCharCodes(canonicalJson.encode(payload));
    1409              :     return true;
    1410              :   }
    1411              : 
    1412            1 :   Future<void> _sendAccept() async {
    1413            2 :     final sas = this.sas = olm.SAS();
    1414            4 :     commitment = _makeCommitment(sas.get_pubkey(), startCanonicalJson);
    1415            3 :     await request.send(EventTypes.KeyVerificationAccept, {
    1416            1 :       'method': type,
    1417            1 :       'key_agreement_protocol': keyAgreementProtocol,
    1418            1 :       'hash': hash,
    1419            1 :       'message_authentication_code': messageAuthenticationCode,
    1420            1 :       'short_authentication_string': authenticationTypes,
    1421            1 :       'commitment': commitment,
    1422              :     });
    1423              :   }
    1424              : 
    1425            1 :   bool _handleAccept(Map<String, dynamic> payload) {
    1426              :     if (!knownKeyAgreementProtocols
    1427            2 :         .contains(payload['key_agreement_protocol'])) {
    1428              :       return false;
    1429              :     }
    1430            2 :     keyAgreementProtocol = payload['key_agreement_protocol'];
    1431            2 :     if (!knownHashes.contains(payload['hash'])) {
    1432              :       return false;
    1433              :     }
    1434            2 :     hash = payload['hash'];
    1435              :     if (!knownHashesAuthentificationCodes
    1436            2 :         .contains(payload['message_authentication_code'])) {
    1437              :       return false;
    1438              :     }
    1439            2 :     messageAuthenticationCode = payload['message_authentication_code'];
    1440            1 :     final possibleAuthenticationTypes = _intersect(
    1441            1 :       knownAuthentificationTypes,
    1442            1 :       payload['short_authentication_string'],
    1443              :     );
    1444            1 :     if (possibleAuthenticationTypes.isEmpty) {
    1445              :       return false;
    1446              :     }
    1447            1 :     authenticationTypes = possibleAuthenticationTypes;
    1448            2 :     commitment = payload['commitment'];
    1449            2 :     sas = olm.SAS();
    1450              :     return true;
    1451              :   }
    1452              : 
    1453            1 :   Future<void> _sendKey() async {
    1454            3 :     await request.send('m.key.verification.key', {
    1455            2 :       'key': sas!.get_pubkey(),
    1456              :     });
    1457              :   }
    1458              : 
    1459            1 :   void _handleKey(Map<String, dynamic> payload) {
    1460            2 :     theirPublicKey = payload['key'];
    1461            3 :     sas!.set_their_key(payload['key']);
    1462              :   }
    1463              : 
    1464            1 :   bool _validateCommitment() {
    1465            3 :     final checkCommitment = _makeCommitment(theirPublicKey, startCanonicalJson);
    1466            2 :     return commitment == checkCommitment;
    1467              :   }
    1468              : 
    1469            1 :   Uint8List makeSas(int bytes) {
    1470              :     var sasInfo = '';
    1471            2 :     if (keyAgreementProtocol == 'curve25519-hkdf-sha256') {
    1472              :       final ourInfo =
    1473            7 :           '${client.userID}|${client.deviceID}|${sas!.get_pubkey()}|';
    1474              :       final theirInfo =
    1475            6 :           '${request.userId}|${request.deviceId}|$theirPublicKey|';
    1476              :       sasInfo =
    1477            7 :           'MATRIX_KEY_VERIFICATION_SAS|${request.startedVerification ? ourInfo + theirInfo : theirInfo + ourInfo}${request.transactionId!}';
    1478            0 :     } else if (keyAgreementProtocol == 'curve25519') {
    1479            0 :       final ourInfo = client.userID! + client.deviceID!;
    1480            0 :       final theirInfo = request.userId + request.deviceId!;
    1481              :       sasInfo =
    1482            0 :           'MATRIX_KEY_VERIFICATION_SAS${request.startedVerification ? ourInfo + theirInfo : theirInfo + ourInfo}${request.transactionId!}';
    1483              :     } else {
    1484            0 :       throw Exception('Unknown key agreement protocol');
    1485              :     }
    1486            2 :     return sas!.generate_bytes(sasInfo, bytes);
    1487              :   }
    1488              : 
    1489            1 :   Future<void> _sendMac() async {
    1490              :     final baseInfo =
    1491           11 :         'MATRIX_KEY_VERIFICATION_MAC${client.userID!}${client.deviceID!}${request.userId}${request.deviceId!}${request.transactionId!}';
    1492            1 :     final mac = <String, String>{};
    1493            1 :     final keyList = <String>[];
    1494              : 
    1495              :     // now add all the keys we want the other to verify
    1496              :     // for now it is just our device key, once we have cross-signing
    1497              :     // we would also add the cross signing key here
    1498            3 :     final deviceKeyId = 'ed25519:${client.deviceID}';
    1499            1 :     mac[deviceKeyId] =
    1500            4 :         _calculateMac(encryption.fingerprintKey!, baseInfo + deviceKeyId);
    1501            1 :     keyList.add(deviceKeyId);
    1502              : 
    1503            6 :     final masterKey = client.userDeviceKeys[client.userID]?.masterKey;
    1504            1 :     if (masterKey != null && masterKey.verified) {
    1505              :       // we have our own master key verified, let's send it!
    1506            2 :       final masterKeyId = 'ed25519:${masterKey.publicKey}';
    1507            1 :       mac[masterKeyId] =
    1508            3 :           _calculateMac(masterKey.publicKey!, baseInfo + masterKeyId);
    1509            1 :       keyList.add(masterKeyId);
    1510              :     }
    1511              : 
    1512            1 :     keyList.sort();
    1513            3 :     final keys = _calculateMac(keyList.join(','), '${baseInfo}KEY_IDS');
    1514            3 :     await request.send('m.key.verification.mac', {
    1515              :       'mac': mac,
    1516              :       'keys': keys,
    1517              :     });
    1518              :   }
    1519              : 
    1520            1 :   Future<void> _processMac() async {
    1521            1 :     final payload = macPayload!;
    1522              :     final baseInfo =
    1523           11 :         'MATRIX_KEY_VERIFICATION_MAC${request.userId}${request.deviceId!}${client.userID!}${client.deviceID!}${request.transactionId!}';
    1524              : 
    1525            3 :     final keyList = payload['mac'].keys.toList();
    1526            1 :     keyList.sort();
    1527            2 :     if (payload['keys'] !=
    1528            3 :         _calculateMac(keyList.join(','), '${baseInfo}KEY_IDS')) {
    1529            0 :       await request.cancel('m.key_mismatch');
    1530              :       return;
    1531              :     }
    1532              : 
    1533            5 :     if (!client.userDeviceKeys.containsKey(request.userId)) {
    1534            0 :       await request.cancel('m.key_mismatch');
    1535              :       return;
    1536              :     }
    1537            1 :     final mac = <String, String>{};
    1538            3 :     for (final entry in payload['mac'].entries) {
    1539            2 :       if (entry.value is String) {
    1540            3 :         mac[entry.key] = entry.value;
    1541              :       }
    1542              :     }
    1543            3 :     await request.verifyKeysSAS(mac, (String mac, SignableKey key) async {
    1544            1 :       return mac ==
    1545            1 :           _calculateMac(
    1546            1 :             key.ed25519Key!,
    1547            2 :             '${baseInfo}ed25519:${key.identifier!}',
    1548              :           );
    1549              :     });
    1550              :   }
    1551              : 
    1552            1 :   String _makeCommitment(String pubKey, String canonicalJson) {
    1553            2 :     if (hash == 'sha256') {
    1554            1 :       final olmutil = olm.Utility();
    1555            2 :       final ret = olmutil.sha256(pubKey + canonicalJson);
    1556            1 :       olmutil.free();
    1557              :       return ret;
    1558              :     }
    1559            0 :     throw Exception('Unknown hash method');
    1560              :   }
    1561              : 
    1562            1 :   String _calculateMac(String input, String info) {
    1563            2 :     if (messageAuthenticationCode == 'hkdf-hmac-sha256') {
    1564            2 :       return sas!.calculate_mac(input, info);
    1565              :     } else {
    1566            0 :       throw Exception('Unknown message authentification code');
    1567              :     }
    1568              :   }
    1569              : }
    1570              : 
    1571              : const _emojiMap = [
    1572              :   {
    1573              :     'emoji': '\u{1F436}',
    1574              :     'name': 'Dog',
    1575              :   },
    1576              :   {
    1577              :     'emoji': '\u{1F431}',
    1578              :     'name': 'Cat',
    1579              :   },
    1580              :   {
    1581              :     'emoji': '\u{1F981}',
    1582              :     'name': 'Lion',
    1583              :   },
    1584              :   {
    1585              :     'emoji': '\u{1F40E}',
    1586              :     'name': 'Horse',
    1587              :   },
    1588              :   {
    1589              :     'emoji': '\u{1F984}',
    1590              :     'name': 'Unicorn',
    1591              :   },
    1592              :   {
    1593              :     'emoji': '\u{1F437}',
    1594              :     'name': 'Pig',
    1595              :   },
    1596              :   {
    1597              :     'emoji': '\u{1F418}',
    1598              :     'name': 'Elephant',
    1599              :   },
    1600              :   {
    1601              :     'emoji': '\u{1F430}',
    1602              :     'name': 'Rabbit',
    1603              :   },
    1604              :   {
    1605              :     'emoji': '\u{1F43C}',
    1606              :     'name': 'Panda',
    1607              :   },
    1608              :   {
    1609              :     'emoji': '\u{1F413}',
    1610              :     'name': 'Rooster',
    1611              :   },
    1612              :   {
    1613              :     'emoji': '\u{1F427}',
    1614              :     'name': 'Penguin',
    1615              :   },
    1616              :   {
    1617              :     'emoji': '\u{1F422}',
    1618              :     'name': 'Turtle',
    1619              :   },
    1620              :   {
    1621              :     'emoji': '\u{1F41F}',
    1622              :     'name': 'Fish',
    1623              :   },
    1624              :   {
    1625              :     'emoji': '\u{1F419}',
    1626              :     'name': 'Octopus',
    1627              :   },
    1628              :   {
    1629              :     'emoji': '\u{1F98B}',
    1630              :     'name': 'Butterfly',
    1631              :   },
    1632              :   {
    1633              :     'emoji': '\u{1F337}',
    1634              :     'name': 'Flower',
    1635              :   },
    1636              :   {
    1637              :     'emoji': '\u{1F333}',
    1638              :     'name': 'Tree',
    1639              :   },
    1640              :   {
    1641              :     'emoji': '\u{1F335}',
    1642              :     'name': 'Cactus',
    1643              :   },
    1644              :   {
    1645              :     'emoji': '\u{1F344}',
    1646              :     'name': 'Mushroom',
    1647              :   },
    1648              :   {
    1649              :     'emoji': '\u{1F30F}',
    1650              :     'name': 'Globe',
    1651              :   },
    1652              :   {
    1653              :     'emoji': '\u{1F319}',
    1654              :     'name': 'Moon',
    1655              :   },
    1656              :   {
    1657              :     'emoji': '\u{2601}\u{FE0F}',
    1658              :     'name': 'Cloud',
    1659              :   },
    1660              :   {
    1661              :     'emoji': '\u{1F525}',
    1662              :     'name': 'Fire',
    1663              :   },
    1664              :   {
    1665              :     'emoji': '\u{1F34C}',
    1666              :     'name': 'Banana',
    1667              :   },
    1668              :   {
    1669              :     'emoji': '\u{1F34E}',
    1670              :     'name': 'Apple',
    1671              :   },
    1672              :   {
    1673              :     'emoji': '\u{1F353}',
    1674              :     'name': 'Strawberry',
    1675              :   },
    1676              :   {
    1677              :     'emoji': '\u{1F33D}',
    1678              :     'name': 'Corn',
    1679              :   },
    1680              :   {
    1681              :     'emoji': '\u{1F355}',
    1682              :     'name': 'Pizza',
    1683              :   },
    1684              :   {
    1685              :     'emoji': '\u{1F382}',
    1686              :     'name': 'Cake',
    1687              :   },
    1688              :   {
    1689              :     'emoji': '\u{2764}\u{FE0F}',
    1690              :     'name': 'Heart',
    1691              :   },
    1692              :   {
    1693              :     'emoji': '\u{1F600}',
    1694              :     'name': 'Smiley',
    1695              :   },
    1696              :   {
    1697              :     'emoji': '\u{1F916}',
    1698              :     'name': 'Robot',
    1699              :   },
    1700              :   {
    1701              :     'emoji': '\u{1F3A9}',
    1702              :     'name': 'Hat',
    1703              :   },
    1704              :   {
    1705              :     'emoji': '\u{1F453}',
    1706              :     'name': 'Glasses',
    1707              :   },
    1708              :   {
    1709              :     'emoji': '\u{1F527}',
    1710              :     'name': 'Spanner',
    1711              :   },
    1712              :   {
    1713              :     'emoji': '\u{1F385}',
    1714              :     'name': 'Santa',
    1715              :   },
    1716              :   {
    1717              :     'emoji': '\u{1F44D}',
    1718              :     'name': 'Thumbs Up',
    1719              :   },
    1720              :   {
    1721              :     'emoji': '\u{2602}\u{FE0F}',
    1722              :     'name': 'Umbrella',
    1723              :   },
    1724              :   {
    1725              :     'emoji': '\u{231B}',
    1726              :     'name': 'Hourglass',
    1727              :   },
    1728              :   {
    1729              :     'emoji': '\u{23F0}',
    1730              :     'name': 'Clock',
    1731              :   },
    1732              :   {
    1733              :     'emoji': '\u{1F381}',
    1734              :     'name': 'Gift',
    1735              :   },
    1736              :   {
    1737              :     'emoji': '\u{1F4A1}',
    1738              :     'name': 'Light Bulb',
    1739              :   },
    1740              :   {
    1741              :     'emoji': '\u{1F4D5}',
    1742              :     'name': 'Book',
    1743              :   },
    1744              :   {
    1745              :     'emoji': '\u{270F}\u{FE0F}',
    1746              :     'name': 'Pencil',
    1747              :   },
    1748              :   {
    1749              :     'emoji': '\u{1F4CE}',
    1750              :     'name': 'Paperclip',
    1751              :   },
    1752              :   {
    1753              :     'emoji': '\u{2702}\u{FE0F}',
    1754              :     'name': 'Scissors',
    1755              :   },
    1756              :   {
    1757              :     'emoji': '\u{1F512}',
    1758              :     'name': 'Lock',
    1759              :   },
    1760              :   {
    1761              :     'emoji': '\u{1F511}',
    1762              :     'name': 'Key',
    1763              :   },
    1764              :   {
    1765              :     'emoji': '\u{1F528}',
    1766              :     'name': 'Hammer',
    1767              :   },
    1768              :   {
    1769              :     'emoji': '\u{260E}\u{FE0F}',
    1770              :     'name': 'Telephone',
    1771              :   },
    1772              :   {
    1773              :     'emoji': '\u{1F3C1}',
    1774              :     'name': 'Flag',
    1775              :   },
    1776              :   {
    1777              :     'emoji': '\u{1F682}',
    1778              :     'name': 'Train',
    1779              :   },
    1780              :   {
    1781              :     'emoji': '\u{1F6B2}',
    1782              :     'name': 'Bicycle',
    1783              :   },
    1784              :   {
    1785              :     'emoji': '\u{2708}\u{FE0F}',
    1786              :     'name': 'Aeroplane',
    1787              :   },
    1788              :   {
    1789              :     'emoji': '\u{1F680}',
    1790              :     'name': 'Rocket',
    1791              :   },
    1792              :   {
    1793              :     'emoji': '\u{1F3C6}',
    1794              :     'name': 'Trophy',
    1795              :   },
    1796              :   {
    1797              :     'emoji': '\u{26BD}',
    1798              :     'name': 'Ball',
    1799              :   },
    1800              :   {
    1801              :     'emoji': '\u{1F3B8}',
    1802              :     'name': 'Guitar',
    1803              :   },
    1804              :   {
    1805              :     'emoji': '\u{1F3BA}',
    1806              :     'name': 'Trumpet',
    1807              :   },
    1808              :   {
    1809              :     'emoji': '\u{1F514}',
    1810              :     'name': 'Bell',
    1811              :   },
    1812              :   {
    1813              :     'emoji': '\u{2693}',
    1814              :     'name': 'Anchor',
    1815              :   },
    1816              :   {
    1817              :     'emoji': '\u{1F3A7}',
    1818              :     'name': 'Headphones',
    1819              :   },
    1820              :   {
    1821              :     'emoji': '\u{1F4C1}',
    1822              :     'name': 'Folder',
    1823              :   },
    1824              :   {
    1825              :     'emoji': '\u{1F4CC}',
    1826              :     'name': 'Pin',
    1827              :   },
    1828              : ];
    1829              : 
    1830              : class KeyVerificationEmoji {
    1831              :   final int number;
    1832            1 :   KeyVerificationEmoji(this.number);
    1833              : 
    1834            4 :   String get emoji => _emojiMap[number]['emoji'] ?? '';
    1835            4 :   String get name => _emojiMap[number]['name'] ?? '';
    1836              : }
        

Generated by: LCOV version 2.0-1